使 用 新 一 代 框架 技术 构建 功能 强大 、 性 能 稳健 的 PHP 站 群 。 
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MVC 是 一 种 先进 的 开发 模式 ， 能 够 解决 团队 开发 之 间 协 同 配合 的 问 
题 ， 使 得 网 站 各 部 件 以 更 高 的 效率 运行 。MVC 模式 将 网 站 分 为 3 大 部 件 ， 
分 别 为 模型 、 视 图 、 欣 制 器 。 这 3 大 部 件 各 目 分 离 ， 但 又 相互 依存 ， 了 最 终 
形成 了 一 个 容易 维护 、 容 易 扩 展 、 高 效 运行 的 网 站 平台 。 对 于 后 台 程 序 
员 ， 借 助 于 MVC 模式 束 可 以 更 加 专注 于 功能 的 实现 ， 而 不 需要 太 多 地 涉 
及 页 面 与 前 端 。 这 种 分 工 协 作 的 最 终 目 的 是 提高 开发 效率 及 项 目 质 量 。 对 
于 个 人 项 目 ， 也 许 在 其 他 编程 技术 〈 例 如 Java, Python) P, MVC 模式 并 
没有 优势 ， 但 在 PHP 中 ， 由 于 支持 混合 编程 ， 所 以 使 用 MVC 模式 进行 编 
程 ， 能 显著 提高 工作 效率 。 

本 书 是 国内 第 一 本 专门 介绍 PHP MVC 开发 模式 的 图 书 ， 全 书 围绕 MVC 
实现 思路 进行 细致 的 讲解 。 通 过 MVC 编程 模式 ， 以 点 带 面 ， 全 面 深入 探讨 
PHP 核心 技术 。 同 时 ， 本 书 也 是 一 本 深入 介绍 利用 PHP 构建 高 性 能 网 站 的 图 
书 ， 通 过 MVC 的 数据 库 中 间 件 ， 可 以 轻松 实现 网 站 和 群体、 谈 写 分 离 等 高 级 应 
用 ， 本 书 在 此 基础 上 还 会 进一步 介绍 当前 流行 的 NoSQL 应 用 、 全 文 搜索 应 用 
等 。 最后， 作者 通过 一 个 上 自行 编写 的 MVC 框架 ， 引 导读 者 开发 属于 自己 的 
PHP MVC 框架 。 

本 书 内 容 通俗 易 恒 、 示 例 形 象 ， 适 合 广大 的 Web 从 业 人 员 阅 读 。 由 于 
PHP 非常 简单 、 易 用 ， 所 以 就 算是 未 接触 过 PHP 的 读者 或 者 初学 者 ， 只 要 
掌握 了 基础 的 面 加 对象 编程 思想 束 可 以 轻松 上 手 。 
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我 的 编程 之 路 

我 本 里 是 学 习 动 画 设 计 的 ， 一 次 侦 然 的 机 会 ， 接 触 到 了 计算 机 编程 。 记 得 那 是 2003 年 
的 时 候 ， 我 刚 拥 有 自己 的 第 一 台 计 算 机 ， 通 过 老师 的 介绍 ， 学 会 了 上 网 。 那 年 暑假 ， 在 广州 
的 计算 机 城 购买 了 一 套 洪 恩 教 育 软 件 ， 认 识 了 网 页 编程 。 那 时 的 网 页 编程 技术 主流 的 是 
ASP， 由 于 ASP 简单 、 易 学 ， 很 快 我 就 使 用 ASP 技术 构建 了 我 的 第 一 个 网 站 (名 称 叫 木棉 
休闲 站 ， 现 已 关 财 )， 我 的 网 站 主要 介绍 天 文 知 识 ， 由 于 那 时 的 ASP 空间 很 贵 ， 而 且 多 数 都 
不 文 持 FSO 组 件 ， 所 以 在 实现 网 片上 传 时 都 是 原 狗 上 传 鸣 《没有 压缩 等 前 期 处 理 )， 随 看 访 
问 人 和 气 的 越 来 越 高 〈 大 约 30 个 请 求 量 )， 很 快 网 站 就 挂 了 。 

后 来 我 应 聘 到 一 家 楼 盘 做 网 络 管理 员 ， 并 日 负责 文案 录入 ， 期 间接 触 到 了 Apache 服务 
器 。 我 发 现 其 网 站 后 台 也 是 使 用 Web 设计 的 ， 可 以 对 图 片 进行 前 期 的 压缩 、 裁 剪 、 加 水 印 
等 操作 ， 而 且 单一 服务 器 竟然 轻松 应 对 1000 个 以 上 请 求 量 ， 而 对 外 提供 查询 服务 的 正 是 
PHP 脚本 。 于 是 每 到 周末 ， 我 束 到 广州 购书 中 心 苗 寻 有 关 PHP 的 书籍 ， 但 是 那 时 的 PHP 资 
料 非常 少 ， 多 数 部 是 关于 JSP LUK ASP 的 。 

2004 年 初 ， 随 着 国内 网 络 的 普及 ， 很 快 网 络 上 就 活跃 了 一 批 开源 爱好 者 ， 他 们 乐意 将 
自己 的 学 习 成 果 分 享 到 网 络 上 ， 也 束 从 那 时 起 ， 我 接触 到 了 PHP。 由 于 早期 的 PHP 在 语法 
上 与 C 很 相似 ， 所 以 我 很 快 就 上 手 了 在 学 校 时 我 业余 学 习 过 C 语言 )。 大 约 学 习 了 3 个 
H, PHP 的 第 见 功能 几乎 能 够 运用 目 如 了 ， 与 很 多 初学 者 一 样 ， 感 觉 PHP 原来 这 么 简单 。 

随 着 Web 2.0 的 到 来 ， 那 时 大 兴 Flash 动画 之 风 ， 我 花 了 9 个 月 时 间 使 用 PHP 十 Flash 技 
术 重 写 了 木棉 网 ( 即 前 木棉 休 亲 站 )， 提 供 了 在 线 制作 大 头 贴 的 功能 ， 并 且 人 允许 将 大 头 贴 肥 
布 到 论坛 及 QQ 空间 上 ， 从 而 有 效 地 积累 了 一 批 用 户 。 也 正 因 为 这 些 经 历 ， 让 我 意识 到 了 设 
计 模 式 的 重要 性 。 

2006 年 ， 为 了 给 木 可 网 添加 一 个 讨论 区 ， 我 化 了 3000 元 钱 购 买 了 一 套 基 于 ASPNET 的 
论坛 程序 。 论 坛 是 搭建 起 来 了 ， 但 是 因为 PHP 与 C# 之 间 的 差异 性 ， 论 坛 并 没有 很 好 地 为 用 
户 提供 服务 《例如 账号 不 同步 ， 帖 子 内 容 不 能 与 文章 内 容 交 互 等 )。 于 是 那 年 5 月 ， 我 决定 
推倒 重 来 ， 兰 心 学 习 C#， 而 这 样 做 的 目的 仅仅 是 为 了 让 系统 账号 同步 。 

由 于 专注 于 学 习 ， 并 没有 多 余 的 精力 管理 网 站 ， 所 以 很 快 网 站 人 和气 就 下 来 了 ， 直 到 
2007 年 ， 遗 憾 地 关闭 了 网 站 。 在 学 习 ASPNET 编程 的 这 段 日 子 里 ， 我 重新 审视 了 PHP 开发 
模式 ， 并 且 将 ASPNET 中 广泛 应 用 的 类 工厂 设计 模式 引入 到 了 PHP 开发 中 。 结 合 Smarty P 
板 引 擎 ， 能 够 很 好 地 实现 分 层 设 计 思 路 ， 这 成 为 我 后 来 开发 PHP 产品 的 主要 设计 模式 。 

2009 年 暑假 ， 我 当时 参与 设计 的 产品 是 基于 Windows CE 的 图 书 管理 系统 ， 终 端 界面 使 
用 C# 技 术 ， 而 后 人 台 惑 是 使 用 熟悉 的 LAMP 组 合 。 其 中 开发 模式 使 用 Zend Framework 编程 
框 染 ， 我 主要 的 工作 就 是 编写 SOAP 服务 ， 当 时 首选 的 技术 方案 是 ASPNET， 因 为 在 
ASP.NET 中 ， 创 建 SOAP 服务 是 最 简单 及 快速 的 方式 ， 但 考虑 到 成 本 及 后 期 扩展 等 原因 ， 节 
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终 放弃 了 该 方案 ， 而 是 使 用 了 Zend Framework 的 SOAP JIREH. IMAAT EE, RA 
PHP MVC 开发 模式 产生 了 浓厚 的 兴趣 ， 我 发 现 PHP MVC 模式 与 之 前 经 常 采 用 的 类 工厂 模 
式 非常 类 似 ， 但 MVC 分 层 更 加 彻底 及 明确 ， 而 且 MVC 框 染 都 是 单一 入 口 的 ， 所 以 无 论 是 
在 调试 还 是 后 期 维护 方面 ， 都 为 程序 员 提 供 了 极 蜗 的 灵活 性 。 为 此 我 试图 寻找 一 球 更 加 适合 
在 中 文 环境 下 使 用 的 MVC 框架 (Zend Frameworkl.x 对 中 文 支持 不 好 )， 所 以 认识 了 国内 比 
较 早 的 FCS CEN ThinkPHP 早期 版 本 )。 

一 转眼 束 过 去 儿 年 了 了 ， 相 信 阅 读本 书 的 读者 与 我 一 样 ， 都 是 出 于 对 编程 的 热爱 才 上 自学 
PHP 的 。 每 个 人 从 目 学 到 所 谓 的 高 手 ， 都 会 经 历 看 一 翻 波 折 ， 有 些 是 必要 的 ， 有 些 则 是 完全 
没 必要 的 。 为 此 我 将 个 人 的 编程 经 历 写 出 来 与 读者 分 享 ， 名望 能 够 帮助 读者 从 中 找到 适合 
己 的 编程 道路 。 总 而 言 之 ， 编 程 是 枯燥 的 ， 但 也 是 快乐 的 ， 当 你 在 编程 中 把 原本 复杂 的 问题 

一 解 开 ， 喜 悦 的 心情 不 用 我 多 言 ， 相 信访 者 也 能 够 感受 得 到 。 

目前 我 主要 从 事 移 动 开 发 及 云 技术 开发 ， 参 与 开发 的 商业 产品 有 “两 物 收 购 站 管理 系 
统 ”“ 海 晨 酒店 管理 系统 ”这 些 产 品 都 是 终端 采用 C#， 后 台 使 用 LAMP 组 合 。 此 外 还 开发 
了 一 个 Windows Phone 休闲 游戏 。 虽 然 工 作 蔷 苗 ， 但 能 够 看 看 目 己 的 代码 服 务 于 社会 ， 这 是 
SZ Loch, 

本 书 的 写作 背景 

当前 市 面 上 很 多 PHP 图 书 ， 都 是 以 介绍 PHP 基础 为 重点 的 ， 例 如 PHP 语法 、 函 数 等 。 
我 身边 的 一 些 有 朋友， 在 阅读 PHP 图 书 时 ， 都 会 发 出 “ 书 怎 么 那么 厚 ? ”的 疑问 。 这 直接 给 
读者 PHP 很 高 深 、 很 难 学 的 印象 。 事 实 上 ，PHP 是 一 门 简 单 易 惜 的 语言 ， 它 的 函数 命名 方 
式 非常 形象 化 ， 只 需要 具备 一 点 瑞 文 基础 ， 环 能够 使 恒 其 合 义 。 值 助 于 主流 的 IDE CLH, ot 
算 没 有 计算 机 编程 基础 的 读者 也 能 够 迅速 上 手 ， 所 以 说 到 底 PH 没有 那么 复杂 。 

当然 ， 并 不 是 说 PHP 的 基础 不 用 学， 事实 上 任何 的 计算 机 编程 技术 ， 入 门 的 课程 还 是 
需要 牢固 掌握 的 ， 例 如 语法 、 结 构 、 运 行 原理 等 。 但 是 ， 由 于 PHP 开源 及 目 由 的 特性 ， 很 
多 开发 者 ， 特 别 是 初学 者 都 没有 开发 模式 的 概念 ， 叶 致 编写 的 代 人 码 在 后 期 维护 时 连 目 己 都 不 
认识 。 造 成 这 一 结果 除了 表面 提 到 的 因素 外 ， 还 由 于 现在 的 PHP 图 书 极 少 介 绍 PHP 的 开发 
模式 。 本 书 内 容 由 始 至 终 ， 一 直 围 绕 独 PHP MVC 展开 ， 利 用 高 效 的 MVC FREIN, WIE 
告别 过 去 “面条 式 ” 的 代码 编写 形式 。 

由 于 PHP 人 全面 开源 的 特性 ， 所 以 现在 国内 外 已 经 出 现 了 很 多 针对 MVC 编程 的 PHP Mr 
架 ， 选 择 哪 一 套 框 架 作为 开发 平台 ， 这 对 于 PHP 开发 人 员 来 讲义 是 一 个 极 具 考 验 的 问题 。 在 
PHP 中 ， 虽 然 也 有 官方 的 MVC 框架 (Zend Framework)， 但 其 影响 力 远 没有 微软 的 ASPNET 
MVC 以 及 Java 的 SSH (Struts+Spring+Hibernate) 那么 大 。 所 以 在 PHP 中 实现 MVC 编程 ， 
小 项 目 通 津 使 用 现 有 的 开源 框架 (包括 Zend Framework) 实现 ;， 而 大 型 项 目 通 稍 采用 目 行 编 
D MVC 框 避 的 方式 实现 〈 或 者 在 现在 的 开源 框 名 上 进行 二 次 开发 )， 本 书 为 了 莱 顾 这 两 种 情 
况 ， 在 内 容 安排 上 均 有 涉及 ， 相 信访 者 能 够 从 中 找到 适合 目 己 的 MVC 框架 。 

Zend Framework 虽然 功能 强大 ， 是 PHP MVC 框架 中 最 具 代 表 的 框架 之 一 ， 很 多 大 型 的 
网 站 都 是 基于 Zend Framework 构建 的 ， 但 是 由 于 种 种 原因 ，Zend Framework 在 国内 应 用 得 
并 不 广泛 ， 加 上 Zend Framework 的 扩展 是 以 组 件 化 的 方式 提供 的 ， 而 一 些 第 三 方 功能 组 件 
在 国内 并 不 能 够 正常 使 用 ， 所 以 本 书 并 没有 以 Zend Framework 作为 PHP MVC 平台 ， 而 是 使 
用 国内 人 气 比 较 局 的 ThinkPHP 作为 MVC 平台 ， 读 者 在 通过 系统 的 学 习 后 ， 不 仅 可 以 对 现 
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有 的 开源 MVC 框架 轻松 上 手 ， 而 且 还 能 动手 编写 一 个 自己 的 MVC 框架 。 

ThinkPHP 是 国内 比较 早 也 是 比较 成 熟 的 PHP MVC 框架 ， 在 实现 MVC 编程 方面 ， 借 鉴 
了 大 量 先 进 的 开源 实现 库 思 路 ， 例 如 Struts, JSP Tag, Smarty 等 。 其 清晰 的 文件 结构 、 高 效 
及 敏捷 的 编程 方式 ， 能 够 让 开发 者 迅速 上 手 。 同 时 基于 其 灵活 的 扩展 机 制 ， 使 得 项 目的 功能 
得 到 无 限 扩 展 。 本 书 将 重点 对 扩展 的 实现 机 制 进行 介绍 ， 例 如 驱动 扩展 、 类 库 扩展 等 ， 这 些 
扩展 的 实现 思路 不 仅 适 用 于 ThinkPHP， 同 样 适用 于 其 他 PHP MVC 框架 。 

本 书 适合 读者 群 

© 各 类 型 的 PHP 程序 员 。 

@ 系统 架构 师 。 

@ 项 目 架 构 师 。 

@ 开源 技术 爱好 者 。 

@ 广大 的 Web 开发 从 业 人 员 。 

@ 计算 机 专业 的 学 生 。 

本 书 主要 内 容 

本 书 共 分 为 3 大 部 分 ， 分 别 为 基础 遍 、 实 战 篇 、 项 目 篇 。 在 内 容 组 织 上 尽量 以 循序 渐进 
的 方式 深入 地 讲解 每 个 知识 要 领 。 初 级 的 PHP 程序 员 在 阅读 本 书 内 容 时 ， 由 于 引用 示例 形 
象 不 会 感觉 生 搬 硬 套 、 敷 衍 应 付 ， 高 级 的 PHP 程序 员 在 阅读 本 书 内容 时 ， 也 会 感受 到 作者 
清晰 的 实现 思路 ， 从 中 获 益 。 本 书 的 内 容 组 织 如 下 。 

第 1 草 主 要 介绍 了 PHP 与 MVC 设计 模式 的 关系 ， 并 且 介 绍 了 在 实验 环境 下 及 生产 环境 
下 的 PHP MVC 运行 环境 搭建 过 程 。 由 于 MVC 设计 模式 的 本 质 是 解决 团队 开发 所 带 来 的 分 
工 问 题 ， 所 以 本 和 草 后 面 还 加 入 了 团队 开发 的 环境 配置 ， 帮 助 每 位 谈 者 建立 起 良好 的 团队 协作 
的 概念 。 

第 2、3 章 主 要 介绍 了 PHP 的 类 结构 。 由 于 后 面 我 们 还 会 重点 介绍 MVC 扩展 的 实现 ， 
以 及 编写 自己 的 MVC HER, MERER E, A SIS PHP 的 面 癌 对 象 编程 是 必要 的 。 
所 以 这 里 安排 了 2 草 针 对 PHP 类 结构 的 内 容 ， 帮 助 读者 加 深 对 类 的 认识 。 

第 4 章 主要 介绍 了 在 PHP 中 主流 的 MVC 框架 发 展 状况 ， 并 通过 简单 的 示例 分 别 演示 了 
这 些 框架 的 简单 使 用 ， 和 帮助 读者 初步 认识 PHP MVC 编程 状况 。 

第 5~9 章 主要 介绍 ThinkPHP MVC 框架 的 使 用 。 这 5 章 内 容 中 ， 作 者 通过 简单 、 形 象 
的 示例 ， 充 分 地 讲解 了 ThinkPHP 在 实现 MVC 编程 中 的 灵活 及 高 效 ， 例 如 MVC 项 目 部 
署 、 数 据 库 CURD 操作 、 功 能 类 库 的 使 用 等 。 此 外 ， 为 了 增强 应 用 的 稳定 性 ， 还 介绍 了 第 
三 方 高 效 的 开源 库 ， 例 如 Nginx 文件 上 传 、HttpSqs 消息 队列 等 。 

第 10、11 草 主 要 介绍 了 ThinkPHP 的 扩展 机 制 。 作 者 将 会 详细 介绍 ThinkPHP REH 
几 种 类 型 扩展 ， 例 如 类 库 扩 展 、 驱 动 扩 展 、 模 型 扩展 、 视 图 扩展 、 行 为 扩展 、 绥 存 扩展 等 。 

第 12 章 将 重点 介绍 PHP 实现 SOAP 服务 的 过 程 ， 使 用 SOAP 服务 ， 束 能 够 实现 高 效 的 
业务 整合 。 

第 13 间 主 要 是 为 了 兼顾 传统 的 PHP 程序 员 更 好 地 利用 ThinkPHP 的 模型 与 控制 器 机 
制 ， 实 现 更 灵活 的 MVC 定制 ， 从 而 将 Smarty 模板 引擎 无 颖 地 移植 到 ThinkPHP 中 。 

第 14 一 16 章 是 专门 为 构建 高 性 能 网 站 而 反 写 的 。 现 在 的 网 站 对 性 能 是 非常 苛刻 的 ， 如 
果 你 的 网 站 正在 被 性 能 所 困扰 ， 建 议 阅 读 这 3 E. 
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第 17 章 是 本 书 内 容 的 浓缩 部 分 ， 作 者 通过 一 个 论坛 系统 项 目 ， 详 细 回 顾 了 本 书 前 面 所 
介绍 的 内 容 精华 ， 帮 助 读者 巩固 所 学 的 知识 。 

第 18 章 再 次 突出 “PHP MVC 开发 ”这 个 主题 ， 通 过 本 章 内 容 读者 将 由 MVC 框架 的 使 
用 者 变 成 MVC 框架 的 开发 者 。 作 者 将 通过 一 个 人 简单 的 MVC 框架 ， 帮 助 读者 了 解 开发 一 个 
MVC 框架 所 需要 的 技术 及 思路 ， 从 而 有 效 提高 PHP 开发 技能 及 组 织 能 力 。 
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内 容 提 要 

MVC 是 计算 机 编程 技术 中 一 种 先进 且 完 善 的 软件 设计 模式 ， 最 早 运 用 在 桌面 软件 开发 
中 。Web 技术 是 近年 来 迅猛 发 展 的 一 种 互联 网 重要 技术 ， 利 用 MVC 设计 模式 代替 传统 的 混 
合式 网 页 编写 技术 ， 不 仅 能 够 极 大 地 提高 代码 质量 ， 还 为 开发 大 型 Web 应 用 提供 可 靠 的 平 


将 会 介绍 PHP 在 实现 MVC 设计 中 的 特性 ， 详 细 介 绍 PHP MVC 设计 模式 的 优 缺 
冬 通过 对 PHP 的 了 解 ， 加 深 对 编程 模式 的 认识 。 本 章 后 面 还 加 入 了 环境 搭建 、 工 具 
容 ， 方 便 读 者 进行 系统 学 习 。 

SES 首先 需要 读者 了 解 Windows 命令 行 操作 和 Linux 终端 的 简单 操作 ， 并 且 拥 有 一 
定 命令 行使 用 习惯 。 

学 习 目标 

© 了 解 PHP 的 历史 、 特 点 及 优点 等 。 

@ 了 解 PHP 与 MVC 之 间 的 关系 。 

@ 全 而 掌握 PHP 各 种 平台 下 的 环境 搭建 。 

@ 认识 PHP 主流 IDE 开发 工具 

@ 全 面 掌握 SVN 版 本 控制 实战 应 用 。 
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1.1 PHP 与 MVC 概述 


PHP 是 一 门 计 算 机 编程 语言 ， MVC 是 一 种 软件 设计 模式 。PHP 可 以 独立 于 任何 平台 
(只 要 该 平台 文 持 PHP 解释 引擎 )， 而 MVC 同样 适用 于 文 持 面 癌 对象 编 程 的 开发 语言 。 事 实 
上 ， 早 期 的 PHP 编程 技术 与 MVC 设计 是 无 缘 的 ， 直 到 PHP 5.x 的 出 现 ，PHP 才 真 正 具备 
MVC 设计 的 概念 。 要 深入 理解 PHP 与 MVC 之 间 的 关系 ， 就 必须 要 对 PHP 的 历史 有 深刻 的 
认识 。 接 下 来 首先 介绍 PHP 的 由 来 ， 然 后 再 介绍 PHP 与 MVC 之 间 的 关系 。 

1. PHP 发 展 历史 

PHP 是 PHP Hypertext Preprocessor 的 简称 ， 从 名 字 上 就 可 以 看 出 它 是 一 种 文本 处 理 程 
序 ， 并 且 是 预 处 理 的 ， 所 谓 的 预 处 理 就 是 在 HTML 中 先入 处 理 脚 本 ， 实 现 功能 运算 。 可 见 
PHP LIZ Jr Web 应 用 而 设计 的 。 

PHP 于 1995 年 诞生 于 Rasmus Lerdorf CHIHI e WER, FZ) ZF, KIRK 
Hr DU Personal Home Page， 是 Rasmus Lerdorf 为 了 完成 个 人 网 站 数据 收集 、 流 量 统计 而 开发 
的 一 种 CGI 工具 集 ，Rasmus Lerdorf 将 窗 体 解释 器 与 表单 数据 进行 集成 ， 并 将 表单 数据 转 
换 ， 实 现 与 数据 库 交 互 ， 称 之 为 PHP/FI (Personal Home Page/Form Interpreter〉 并 将 源码 放 
置 到 开源 社区 借 此 来 加 速 程序 的 开发 与 得 找 错误 。 

1995 年 底 ， 开 源 后 的 PHP/FI 被 正式 命名 为 PHP 2， 该 版 本 已 经 具有 了 现代 语言 的 一 些 
特性 ， 借 鉴 了 Perl 的 许多 特点 ， 使 用 了 Perl 的 变量 命名 方式 、 窗 体 处 理 方式 、HTML 标记 
先入 等 ， 计 PHP 具有 了 更 灵活 的 弹性 。 同 时 Rasmus Lerdorf 使 用 C 语言 重 写 了 PHP/FI 的 编 
译 器 ， 提 供 了 完善 的 数据 库 访问 接口 ， 让 PHP 2 运行 得 更 加 稳定 和 快速 。 

1997 年 ， 两 个 出 色 的 以 色 列 程序 员 Zeev Suraski 和 Andi Gutmans 第 一 次 参与 了 PHP 解 
释 器 的 设计 ， 并 出 色 地 与 Rasmus Lerdorf 合作 ， 完 成 了 PHP 2 的 升级 ， 取 名 为 PHP/FI2， 此 
次 升级 为 PHP 3 的 到 来 真 定 了 方向 〈 是 PHP3 的 雏形 )。 另 外 Zeev Suraski 和 Andi Gutmans 
为 了 消除 一 些 名 称 叫 法 的 混乱 ， 亲 循 开源 社区 协议 ， 将 PHP/FI 2 及 后 续 版 本 统称 为 PHP 
(PHP Hypertext Preprocessor 依旧 保留 )。 

1998 年 ， 在 Zeev Suraski 和 Andi Gutmans 的 努力 工作 下 ， 完 成 了 PH 一 次 重大 的 升 
级 ， 这 次 升级 重 写 了 PHP 解释 器 的 内 核 ， 称 之 为 PHP 3。Zeev Suraski 和 Andi Gutmans 还 在 
以 色 列 成 这 了 PHP 商业 化 运作 公司 Zend Technologies， 并 于 1999 年 发 布 了 Zend Engine 5| 
擎 ， 该 引擎 能 够 为 PHP 这 来 更 加 高 速 与 稳定 的 环境 文 撑 。 

2000 年 ，Zend Technologies 在 Zend Engine 的 基础 上 发 布 了 PHP 4.0。PHP 4.0 是 PHP 的 
一 次 重大 升级 ， 提 供 了 众多 数据 库 接口 、 网 络 函 数 、 文 件 操作 函数 等 ， 使 得 PHP 真正 成 为 
最 主流 和 最 快捷 的 Web 应 用 开发 语言 。 

2004 年 7 H, ， PHP 正 陈 成 为 真正 意义 上 的 现代 化 编程 语言 ， 在 开源 社区 和 Zend 
Technologies 的 努力 下 PHP 5 如 期 而 来 ， 该 版 本 使 PHP 真正 拥有 了 OOP (HEIR) J 
程 概念 ， 使 得 PHP 华丽 地 转 号 为 Java 最 直接 的 对 手 ， 许 多 著名 的 网 站 开始 由 Java Fein 
PHP 5., 

PHP 5 同 Java EREET OOP 概念 、 模 块 概念 ， 并 引入 了 数据 访问 中 间 层 (PDO)。 
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PHP 5 以 前 的 版 本 在 进行 团队 开发 时 显得 力不从心 ， 大 一 点 的 项 目 往往 都 会 抛弃 PHP， 
PHP 5 的 出 现 让 PH 真正 具备 了 企业 级 开发 能 力 ， 并 且 继 承 了 前 面 版 本 中 的 高 效 与 敏捷 ， 兼 
容 旧 版 本 PHP 大 多 数 函 数 和 语法 ， 让 PHP 无 颖 过 渡 ， 得 到 了 Yahoo, Google 等 网 络 巨 头 的 
强力 文 持 ， 使 得 PHP 5 成 为 世界 上 最 流行 的 Web 应 用 开发 语言 。 截 止 笔者 定稿 为 止 ， 当 前 
最 狐 稳 定 版 本 为 PHP 5.3.8。 

下 一 个 版 本 将 迎 来 PHP 史上 最 大 的 改变 ， 根 据 目前 所 综合 的 信息 ， 代 号 为 PHP 6 的 下 
一 代 Web 开发 语言 将 会 支持 多 线程 、 支 持 Unicode (多 国语 言 )、 支 持 Collation 〈 字 符 集 整 
理 )、 文 持 语 言 翻译 等 ， 与 Java 和 (# 一 样 深入 文 持 命 名 空间 (5.3.1 已 经 初步 支持 )， 使 得 
PHP 真正 成 为 现代 化 的 编程 语言 技术 。 另 外 ， 下 一 代 PHP 技术 为 了 提高 运行 速度 与 稳定 
性 ， 也 会 移 除 一 些 功 能 和 函数 (如 register_globals, magic_quotes, safe_mode 等 )。 

PHP 的 发 展 大 体 上 可 分 为 4 个 步骤 : 处 于 闻 牙 时 期 的 1995 一 1998 年 ;处 于 成 长 期 的 
2000 一 2002 年 ; 处 于 成 熟 期 的 2002 一 2005 年 以 及 处 于 稳定 期 的 2008 年 到 现在 。Zend 公司 
从 4.0 版 本 开始 ， 采 用 的 是 双 版 本 更 狐 策 略 ， 如 图 1-1 ra, 

总 而 言 之 ，PHP 是 让 人 兴奋 的 ，PHP 6 更 让 全 球 的 Web 应 用 开发 者 和 企业 期 待 ， 读 者 如 
需 了 解 更 多 关于 下 一 代 PH 技术 ， 可 以 浏览 PH 官方 的 技术 支持 网 站 http://www.php.net， 
获取 最 新 消息 。 

2. PHP 5 MVC 设计 

MVC 是 一 种 设计 模式 ， 在 软件 设计 工程 中 非常 实用 ， 最 早 由 smalltalk 语言 研究 中 心 提 
H, Java 的 壮大 极 大 地 让 定 了 MVC 设计 思想 。 目 前 ， 基 于 Java 的 MVC MHA Struts, 
Spring, Grails 等 ，Java 程序 员 可 以 利用 这 些 框 染 ， 极 大 地 提 咒 开 发 效率 。 

PHP 经 历 了 面向 过 程 、 面 向 函数 到 PHP 5.0 的 面向 对 象 。 多 种 设计 模式 的 演变 ， 使 得 
PHP 变 得 非常 易 用 和 强大 ， 但 是 也 让 PH 始终 没有 进入 现代 化 设计 语言 的 阵营 。PHP 5 后 的 
版 本 借鉴 了 大 量 Java 思想 ， 得 葵 于 早期 PHP 面 癌 过 程 编程 文 持 ，PHP 在 实现 MVC 设计 中 
变 得 较 灵 活 、 易 用 。 首 先 看 一 段 代 码 。 

<html> 

<head> 


<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
<title> 测 试 </title> 


</head> 
<body> 
<div class="main"> 
<ul> 
<1i> 用 户 名 : <?php echo S$rwos["username"];?></1i> 
<11> 密 亿 : <?php echo ŝrwos["password"];?> 
<1i> 姓 别 : <?php 
if(S$rows["sex"])t{ 
eono WH: 
}else{ 
echo "4"; 
EE 
SS 
</ w> 
</div> 
</body> 
</html> 
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1995 年 6 月 8 日 Rasmus Lerdorf 在 Perl 语言 的 基础 上 发 明了 PHP 语言 
1996 年 4 月 16 日 PHP2.0 诞 生 


1998 年 6 月 6 日 Zeev Suraski 和 AndiGutmans 与 Rasmus Lerdorf 合作 ， 
重 写 了 PHP 解释 引擎 ， 并 发 布 PHP 3.0 


2000 年 5 月 22 日 40 发 布 ，PHP 解释 器 被 Zend 解释 器 不 换 


2001 年 12 月 10 日 4.1.0 发 布 ， 提 供 了 超 全 局 变量 ， 
简化 接收 表单 数据 的 过 程 


2002 年 4 月 22 日 4.2.0 默认 取消 register_globals 功能 


2002 年 12 月 27 日 43.0 发 布 ，5 引 人 节令 行 可 执行 文件 功能 


2004 年 7 月 13 日 发 布 5.0.0 全 面 进 人 OOP 时 代 


2005 7 H 11 HH 4.4.0 


2006 Æ 11 月 2 日 发 布 5.2.0 


2008 年 1 月 3 日 发 布 4.4.8 


2008 年 8 月 7 日 发 布 44.9 修补 了 PHP 之 前 的 所 有 漏洞 


2008 年 12 月 8 日 发 布 5.2.8 


2009 年 2 月 26 日 发 布 5.2.9, 改善 了 稳定 性 


2009 年 6 月 30 日 发 布 5.3.0 提供 了 丰富 的 网 络 数 据 处 理 接口 ， 
40 XMLReader 和 XMLWriter、SOAP 等 


2010 年 7 月 22 日 发 布 53.3 提供 命名 空间 编程 


图 1-1 PHP 大 事 记 
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上 述 代码 是 传统 PHP 编程 技术 中 典型 的 代码 编写 方式 。PHP 允许 开发 人 员 和 直接 在 HTML 
网 页 中 檬 入 逻辑 运算 代码 ， 这 种 类 似 于 JavaScript 的 脚本 编写 方式 ， 曾 经 大 受 网 页 开发 人 员 欢 
迎 。 但 是 也 正 因 如 此 ，PHP 5.x 之 前 的 版 本 并 不 能 胜任 大 型 Web 项 目的 构建 ， 因 为 过 多 地 在 页 
面 中 能 入 运算 代码 ， 无 颖 对 项 目 后 期 维护 产生 极 大 的 困扰 。 同 时 ， 由 于 光 辑 与 视图 混在 一 起 ， 
将 造成 团队 协同 的 困难 。 所 以 早期 的 PHP 通 第 只 适用 于 小 型 或 个 人 网 站 。 但 是 随 看 PHP 5.x 
的 到 来 ， 各 种 MVC 框架 的 出 现 ， 尤 其 Zend 推出 的 Zend Framework， 彻 底 地 改变 了 PHP 状 
况 ， 使 得 PHP 也 能 够 实现 优雅 的 Web 编程 。MVC 设计 模式 将 使 代码 将 变 得 简洁 。 


<?php 

8 /二 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 
// | index 控制 器 

/二 = 一 一 一 一 一 一 一 一 一 一 一 一 一 


class IndexController extends Controller { 

public function Index (){ 
suser=MOD ("User"); 
Şrows=$user—->rows () ; 
Sthis->View();}; 
Sthis->smarty->assign(n"npageTit1len," 会 员 中 心 首 页 ") ; 
SE EE Ee Dee D E Se F 
Şthis->smarty->display ("index.html"); 


} 


p> 


如 上 述 代码 所 示 ， 传 统 意义 上 ，Index 是 一 个 PHP 类 的 方法 。 但 是 在 MVC 设计 模式 
H, Index 称 为 动作 。 对 于 普通 用 户 而 言 束 是 一 个 网 页 。 但 是 ， 这 里 的 网 页 并 不 像 前 面 所 讲 
的 网 页 需要 混杂 看 HTML 代码 ， 这 就 是 MVC 设计 模式 中 最 典型 的 前 后 人 台 分 离 编程 特点 。 

当然 ，MVC 只 是 一 种 设计 模式 ， 在 开源 的 PHP 编程 世界 中 ， 己 经 拥有 众多 的 MVC 编 
程 框架 。 尺 管 每 个 框架 在 实现 MVC 方式 上 有 所 不 同 ， 但 无 论 怎样 变化 ， 都 拥有 Model (Ré 
型 )、View〔 视 图 ) 及 Controller GEE) 概念 。 这 三 者 之 间 的 关系 如 图 1-2 所 示 。 


模型 


封装 应 用 程序 状态 

| 啊 应 状态 查询 
通 应 用 程序 功能 
通知 改变 通知 视图 改变 


视图 
解释 模型 
模型 更 新 请 求 


farlar 


定义 应 用 程序 行为 

Ee eg 与 请求 用 户 动作 映射 成 模型 更 新 
发 运用 户 输入 给 控制 器 See e 选择 响应 的 视图 
允许 控制 右 选 择 视 图 


图 1-2 MVC 运行 图 


如 图 1-2 HR, Pia WE AEX 3 个 部 分 都 是 相互 分 离 又 互相 依存 的 ， 它 们 具备 
较 高 的 烛 合 性 ， 控 制 问 是 MVC 中 的 指挥 员 ， 模 型 右 担 当 了 MVC 的 动力 源 ， 模 型 大 可 以 连 
接 到 传统 的 PHP 类 库 ， 开 发 人 员 也 可 以 使 用 现 有 的 函数 库 等 进行 扩展 。 

综合 而 言 ，MVC 编程 的 灵魂 就 是 灵活 。MVC 设计 不 像 传统 的 “类 工厂 ”模式 只 在 有 限 
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的 类 库 中 进行 功能 扩展 ; MVC 能 够 根据 用 户 的 请 求 ， 由 控制 器 指挥 模型 占 进 行 相应 的 算 
法 ， 得 出 的 结果 由 视 网 解 释 需 进行 呈现 。 
3. 使 用 PHP MVC 的 优 缺 点 
通过 前 面 的 介绍 ， 相 信 读 者 已 经 能 够 对 PHP 与 MVC 设计 有 了 初步 理解 。 使 用 PHP 
MVC 开发 模式 的 显著 优点 如 下 。 
> 利用 MVC 框架 提供 的 数据 库 操作 中 间 层 ， 能 够 蜗 效 、 安 全 地 对 各 种 数据 库 进行 
操作 。 
> MVC 提供 了 先进 友好 的 前 台 与 后 侣 分离 功能 ， 使 得 一 个 团队 中 界面 设计 人 员 与 后 台 
编程 人 员 更 好 、 更 高 效 地 协作 。 
> MVC 的 蜗 度 灵活 性 ， 能 够 在 程序 开 友 的 任何 阶段 增强 的 层 类 库 ， 使 得 在 不 修改 或 少 
量 修改 项 目 源 代 码 的 基础 上 ， 实 现 更 强大 的 功能 。 
> MVC 框架 都 文 持 数据 及 文件 缓存 ， 并 文 持 代 码 预 编 详 《〈 部 分 MVC MERLI), 1 
程序 代 人 码 运行 效率 更 加 高 效 。 
> MVC 框架 从 底层 代码 入 口 ， 对 所 有 POST 及 GET 提交 均 会 做 安全 过 滤 ， 所 以 基于 
MVC 编写 的 网 站 都 能 够 得 到 很 好 的 安全 保护 。 
> MVC 对 网 站 URL 访问 进行 了 优化 ， 有 效 地 改善 用 户 体 验 。 
当然 ，MVC 设计 并 非 上 只 有 优点 ， 也 存在 缺点 ， 下 面 将 进行 简单 总 结 。 
> 对 于 从 单 的 界面 ， 严 格 壮 循 MVC， 使 模型 、 视 图 与 控制 名 分 离 ， 会 增加 结构 的 复杂 
性 ， 并 可 能 产生 过 多 的 更 新 操作 ， 降 低 运 行 效 率 。 
> 视图 与 控制 器 是 相互 分 离 ， 但 却 是 联系 紧密 的 部 件 ， 视 图 没有 控制 器 的 存在 ， 其 应 
用 是 很 有 限 的 ， 反 之 亦 然 ， 这 样式 妨碍 了 它们 的 独立 重用 。 
> 依据 模型 操作 接口 的 不 同 ， 视 图 可 能 需要 多 次 调用 才能 获得 是 够 的 显示 数据 。 对 未 
变化 数据 的 不 必要 的 频繁 访问 ， 也 将 损害 操作 性 能 。 
> 缺少 针对 性 的 PHPIDE 支持 (Zend Framework 除外 )。 
任何 东西 都 是 具备 两 面 性 的 ， 尤 其 在 计算 机 编程 中 更 是 如 此 。MVYC 固然 有 其 缺点 ， 但 
其 带 来 的 好 处 远 超 其 缺点 。 尤 其 对 于 大 型 Web 应 用 开发 来 说 ， 更 能 显示 出 MVC 开发 模式 的 
巨大 优势 。 本 书 束 是 一 本 专门 针对 PHP MVC 设计 模式 的 图 书 ， 不 仅 全 面 介 绍 MVC 实战 内 
容 ， 最 后 还 将 介绍 MVC 模式 的 实现 方式 。 


1.2 ”开发 环境 搭建 


前 面 对 PHP 技术 与 PHP 发 展 情况 作 了 一 些 详 细 的 介绍 ， 要 运行 PHP 必须 要 安装 与 配置 
PHP 所 需要 的 运行 环境 。 本 节 将 会 介绍 PHP 运行 环境 的 搭建 ， 让 读者 特别 是 初次 接触 PHP 
的 读者 能 够 跑 起 第 一 个 PHP 应 用 程序 。 

PHP 运行 环境 通常 分 为 两 种 情况 . 一 种 是 用 于 真实 生产 环境 的 ， 另 一 种 用 于 开发 或 演示 
环境 的 。 对 于 真实 的 生产 环境 需要 做 较 多 的 工作 ， 需 要 结合 操作 系统 的 安全 策略 和 特点 详细 
配置 PHP 每 个 模块 ， 考 虑 到 成 本 与 性 能 因素 通常 都 是 在 Linux 平台 上 搭建 的 。 相 比较 而 言 开 
发 环境 下 的 PHP 运行 环境 比较 简单 ， 开 发 人 员 只 需要 安装 Web 服务 器 和 PHP 解释 引擎 ， 即 
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可 正式 进入 开发 状态 。 

目前 生产 环境 上 的 平台 使 用 最 多 的 是 LAMP (Linux+Apache+MySQL+PHP) 组合; 以 
及 被 新 浪 、 淘 宝 等 国内 外 大 企业 广 为 采用 的 LNMP 〈Linux+Nginx+MySQL+PHP) 组 合 。 后 
者 使 用 HTTP 反 向 代理 ， 能 够 提供 极 具 速 度 优势 、 系 统 资 源 占 用 少 的 特点 ， 近 年 来 已 经 三 沁 
地 被 世界 上 超大 型 的 网 站 所 接受 ， 大 有 超越 Apache 的 趋势 。 与 Apache 相 比 ， 它 体积 非常 
小 ， 功 能 模块 也 就 少 得 多 。 需 要 注意 的 是 ， 无 论 是 LAMP 组 合 还 是 LNMP 组 合 ， 当 中 的 M 
代表 MySQL， 通 党 情况 下 MySQL 都 是 使 用 独立 服务 器 《服务 器 集群 ) 进行 安装 的 。 这 里 
所 说 的 PHP 环境 搭建 是 指 开 发 环境 的 搭建 ， 开 发 环境 的 搭建 义 分 为 Windows 与 Linux WA 
平台 ， 下 面 分 别 对 这 两 种 平台 的 PHP 安装 过 程 作 一 个 详细 的 讲解 。 


1.2.1 在 Windows 下 使 用 一 键 安装 


在 Windows 平台 上 开发 PHP 应 用 是 最 常见 的 ，Windows 出 色 的 图 形 界面 能 够 极 大 地 提 
高 PHP 应 用 程序 的 开发 速度 ， 为 此 一 些 PHP 爱好 者 和 组 织 也 开发 了 Windows 下 的 PHP 一 
键 安装 包 ， 使 用 这 些 安 装 包 就 能 够 轻易 地 完成 PHP 环境 的 搭建 ， 如 PHPnow、phpStudy、 
XAMPP 等 。 出 于 易 用 性 及 效率 考虑 ， 本 节 将 会 详细 讲解 Windows 平台 的 XAMPP 一 键 安装 
包 ， 使 用 一 键 安 北 包 除了 安全 性 差 之 外 ， 其 他 的 功能 和 真实 的 生产 环境 是 一 样 的 。 

XAMPP 原名 是 LAMPP， 由 德国 人 开发 而 成 ， 近 年 来 广泛 地 被 开发 者 所 接受 ， 最 
重要 的 一 点 束 是 它 非 党 容易 安装 和 使 用 ， 并 且 集 成 了 PHP 众多 模块 ， 几 乎 所 有 主流 的 
PHP 扩展 都 已 被 内 置 。XAMPP 具有 容易 使 用 、 高 效 开 发 和 稳定 运行 的 特点 ， 已 经 被 集 
成 到 了 Eclipse PDT 中 ， 这 样 开 发 人 员 就 可 以 利用 Eclipse 的 强大 调试 功能 ， 实 现 PHP 
的 敏捷 开发 。 

(1) 下 载 XAMPP 

Hui XAMPP 首先 需要 获得 XAMPP 安装 包 ， 读 者 可 以 前 往 XAMPP 的 下 载 幢 面 进行 
下 载 ， 地 址 为 http://www.apachefriends.org/zh_cn/xampp-windows.html，XAMPP 安装 方式 共 分 
为 两 种 : 一 种 为 使 用 Windows 软件 包 安 装 方式 ; 男 一 种 为 直接 编 译 源 代码 ， 即 绿色 安装 。 
两 种 方式 同样 简单 ， 如 果 使 用 Windows 软件 包 的 安装 方式 ， 在 安 冯 完 成 后 操作 系统 会 记录 
此 次 安装 的 过 程 并 反馈 结果 ， 然 后 局 动 相应 的 服务 模块 ， 最 后 在 开始 末 单 中 出 现 XAMPP 可 
视 化 管理 面板 ， 下 面 分 别 介 绍 。 

XAMPP 当前 最 新 版 本 为 1.7.7。 找 到 网 页 中 的 “XAMPP for Windows 1.7.7, 20.9.2011”, 
My “Installer” PE, WA Windows 软件 安装 包 的 XAMPP FREER, Jul 1-3 所 示 。 


Version Size Content 


XAMPP Windows 1.7.7 Apache 2.2.21; MySQL 5.5.16, PHP 5.3.6, OpenSSL 1.0.0. 
phpMyAdmin 3-4.5, XAMPP Control Panel 2-5, Webalizer 
2.233-04, Mercury Mail Transport System v4.72, FileZilla FTF 
Server 0.9.39, Tomeat 7.0.21 (wth mod_proxy_ajp as 
connector) 
For Windows 2000, XP, Vista, 7. 


Installer 81 MB 安装 包 

MDS checksum: 45008684a3bd21343fr69fcef2f4577 be 
7 ZIP 149 MB ZIP ARE 

MOS checksum: 19c856c350f73a19f049d65126367f0e 
EI 69 MB 7zip FB 


MOS checksum: fëcäcebäaidaADët1ëdeggieaëfëpatog 


图 1-3 ”下载 XAMPP 
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按照 回 导 页 面 提示 ， 完 成 XAMPP 的 下 载 。 
(2) 安装 XAMPP 
XAMPP 的 安装 非常 简单 ， 双 击 下 载 的 安装 包 ，XAMPP 安装 癌 导 将 局 动 ， 如 图 1-4 所 


lege XAMPP 界面 语言 上 只 文 持 英语 ， 单 击 “OK” 按 钮 ，XAMPP 安装 向 导 将 会 进 
闭 人 确认 状态 ， 如 图 1-5 所 示 。 


Space avalable: 58.868 


Nullsoft Instal System v2.35 


图 1-4 XAMPP 安装 路 径 图 1-5 XAMPP 安装 向 导 


确认 安装 路 径 后 ， 一 直 点 击 “Next>” 按 钮 ， 直 到 安装 完成 。 在 安装 完成 后 可 以 在 “ 开 
台 ” 菜 单 下 找到 XAMPP 的 管理 面板 ， 启 动 后 如 图 1-6 所 示 。 


Le e 
[E] XAMPP Control Panel Application “OO c x 


XAMPP Control Panel [Esc | 
Modules 
| 回 sve Apache Running Leg 
Esve  MySal Running 
Flsve FileZilla 
[Sve Mercury 


facian Directory: CEC:" Xampp 

| Install (ler) Directory: Champ 
| Busy- - 

[apache atarted [Port Go) 
Big... 

IMySgql started [Port 3306] 


图 1-6 XAMPP 管理 面板 


在 XAMPP 管理 面板 中 可 以 对 各 服务 组 件 进行 管理 ， 例 如 “Start” “Stop” S. WRT 
要 随机 启动 ， 在 相应 组 件 的 Sve EFT “v” BNET. 

表面 讲述 的 是 使 用 安装 包 的 方式 进行 安 逆 ， 和 其 他 Windows ZRI RA ZAKA 
同 ， 读 者 应 该 能 够 迅速 掌握 。 (ZIP 包 ) 进行 环境 配置 。 

使 用 源码 包 进 行 安装 显得 更 加 干净 和 快捷 ， 读 者 在 安 逆 时 ， 只 需要 释放 源 代 码 包 到 磁盘 
下 的 某 一 目录 ， 甚 至 解压 到 U 盘 ， 即 可 初始 化 XAMPP。 
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ZIP 源 代码 包 相 对 Installer 安装 包 来 说 文件 体积 大 了 许多 ， 解 压 后 会 在 目录 下 看 到 
XAMPP 的 文件 组 成 结构 ， 如 网 1-7 所 示 。 


xampp (39) 

DADownloads\wampp-win32-1.7.A4-VO6 

bk anonymous 局 sre 区 | mysql_stoP,bat 
k apache L tmp E| passwords.txt 

k cgi-bin | tomcat E| readrme de tt 

|i contrib [i webalizer 后 | readme_en.txt 

k FileaillaFTP webdav Eservice.exe 

li htdocs Gë apache start bat [区]|setup xampp.bat 
bk install 2. apache stop.bat [Bjxampp starteve 
[i licenses 2. catabna start bat [=]jxampp_stop.exe 
I MercuryMail El catalina_stop.bat [E]xampp-controlexe 
[i mysq! 区 |filezilla_setup.bat 

ji peri 2 Dlezälla start bat 

|i php lfilezilla_stop.bat 

[i phpMyådmin {| mercury_start.bat 

|i security El mercur Stop, bat 

[i sendmail mell start bat 


图 1-7 XAMPP 源 代码 包 组 成 文件 


双击 文件 夹 下 的 “setup-xampp.bat” 批 处 理 文 件 ， 将 会 启动 XAMPP 初始 化 配置 程序 ， 
如 图 1-8 所 示 。 初 始 化 配置 文件 完成 后 ， 此 时 就 可 以 打开 目录 下 的 “xampp_stop.exXe” 文 
(F, ZBA XAMPP 的 管理 面板 ， 乞 的 使 用 方式 和 使 用 Installer 安装 包 的 方式 是 一 样 
的 ， 机 和 面 已 经 介绍 过 ， 在 此 不 再 重 述 。 


国 C\Windows\system32\cmd.exe 


EEE E EEE EEE EE EEE E E HHH H HRH HHH HH HRH H HHH EEE E EEE HRHHH HHH HRH HHH H HHH 

H ApacheFriends XAMPP setup win32 Version H 

H H 

H Copyright Ceci 2002-2009 fpachefriends 1.7.4 H 

H H 

H Authors: Kay Uogelgesang <kvofapachefriends.org> H 
! H 


H Carsten Miednann Zveboacterbäuiednann-online de 3 


UREE E EEE EEEE EEE EEEE E RER EEE EEE E EEEE EEE EEE E CEEE EEE E HR HRHH RH HRH H 


Sorry, but ... nothing to do?! 
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图 1-8 XAMPP 初始 化 


在 管理 面板 中 局 动 相 关 的 组 件 后 ， 打 开 浏 览 右 进入 http://localhost 网 址 ， 如 果 浏 览 结 
如 图 1-9 所 示 ， 证 明 XAMPP CARRI, PHP 所 需要 的 开发 环境 已 经 焉 绪 。 

如 图 1-9 所 示 ， 读 者 可 以 单 击 右 侧 导航 栏 中 的 功能 链接 ， 检 查 Apache 和 PHP 的 运行 状 
态 ， 例 如 单 击 “perlinfo0” 连 接 ， 将 会 进入 perlinfo0 检 查 状 态 ， 如 图 1-10 所 示 。 

XAMPP 提供 一 键 式 安装 ， 能 够 完成 常见 PHP 组 件 的 安装 ， 拥 有 完善 的 管理 面板 ， 非 常 
易于 开发 环境 的 搭建 。 男 外 XAMPP 还 能 文 持 PHP5 与 PHP4 的 切换 ， 但 由 于 本 书后 面 的 内 
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vest MVC 的 ， 所 以 如 果 读 者 需要 使 用 XAMPP 来 作为 开发 环境 ， 一 定 需要 确保 PHP 的 
行 环境 为 PHP 5.1 以 上 版 本 (默认 状态 即 可 )。 


-C XAMPP for Windows me 


1-9 XAMPP 运行 结果 


This is perl, vë 101 buik for MSWin32-x86-multi-thread 
Copyright (c) 1987-2011, Larry Wall 


1-10 ”查看 PHP 状态 


D 说 明 : XAMPP 源 代 码 包 (ZP 包 ) 不 要 解压 到 中 文 名 称 目 录 下 ， 人 否则 将 会 造成 MySQL 组 件 局 动 失 
败 。 另 外 如 果 本 机 上 已 经 安装 了 其 他 Web 服务 器 (如 HS) ， 通 常情 况 下 都 会 被 预先 占用 80 端口 ， 在 
初始 化 XAMPP 前 ， 最 好 将 其 他 Web 服务 器 暂停 ， 或 者 修改 服务 器 的 默认 端口 。 


1.2.2 在 Linux 平台 安装 LNMP 


XAMPP 是 世界 上 使 用 最 多 的 PH 开发 环境 集成 安装 包 ， 同 样 提供 了 Linux ZRA, H 
于 它 的 安装 过 程 非常 简单 ， 只 需要 儿 个 命令 即 可 ， 在 此 束 不 进行 过 多 的 介绍 ， 读 者 如 需 了 解 
更 多 ， 可 以 浏 贤 官 方 的 双 持 网 页 http://www.apachefriends.org/zh_cn/xampp-linux.html#1673。 
接 下 来 将 介绍 以 性 能 优越 而 著称 的 LNMP 环境 。 
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1. Nginx 

所 谓 的 LNMP E} Linux+Nginx+MySQL+PHP 软件 包 的 组 合 名 。 对 于 MySQL 及 PHP 相 
信 读 者 已 经 非 第 熟悉 ，Nginx 是 最 近 两 年 才 在 国内 流行 的 ， 为 了 方便 后 和 面 的 笠 习 ， 接 下 来 首 
先 对 Nginx 进行 简单 介绍 。 

(C1) Nginx 的 简介 

Nginx 是 由 俄罗斯 综合 门户 网 站 Rambler.ru 开发 并 维护 的 一 套 Web 服务 代理 软件 ， 在 应 
对 大 访问 量 的 情况 下 ， 是 Apache WEIER Nm. Nginx 之 所 以 性 能 局 效 ， 主 要 由 于 Nginx 
采用 了 epoll 网 络 模型 ， 访 模型 是 Linux 2.6 内 核 中 新 的 IO 接口 ， 能 够 有 效 地 减少 CPU 性 能 
消耗 。 最 为 重要 的 一 点 是 Nginx 本 里 只 文 持 纯 静 态 的 文件 解释 。 对 于 动态 脚本 ，Nginx 不 像 
Apache 直接 加 载 模 块 进行 解释 ， 而 是 将 请 求 所 区 给 CGI 进程 管理 右 《〈 例 如 PHP-FPM)， 这 
种 分 工 明 确 但 配合 案 密 的 工作 方式 ， 使 得 Nginx 时 刻 保持 稳定 、 高 速 的 运行 状态 。 

由 于 Nginx 出 色 的 表现 ， 世 界 上 许多 大 型 网 站 均 使 用 Nginx 来 作为 负载 均衡 服务 右 、 绥 
FERIA Web 服务 器 等 。 国 内 的 淘宝 网 是 最 早 使 用 Nginx 的 中 文 网 站 ， 甚 全 在 Nginx 的 基 
础 上 开发 了 Tengine 项 目 ， 感 兴趣 的 读者 可 以 浏览 项 目 网 址 http://tengine.taobao.org/。 

对 于 普通 的 管理 人 员 或 者 PHP 程序 员 而 言 ， 使 用 Nginx 比 使 用 Apache、Lighttp 更 加 容 
SA. Nginx 的 配置 文件 简单 明了 ， 并 不 需要 太 多 的 额外 知识 即 可 掌握 。Nginx 文 持 Apache 主 
流 的 功能 ， 甚 至 支持 得 更 好 ， 包 括 UrlRewWriter、 数 据 缓存 、 信 息 收 集 、FastCGI、CGI 等 。 
下 面 将 首先 讲解 在 CentOS 6.0 操作 系统 下 安装 Nginx 的 全 过 程 。 

(2) Nginx HJ z3% 

首先 以 Root 的 映 份 登录 CentOS, EAIA FHA “mkdir -p /datal ”命令 创建 数据 存 
放 目 录 ， 然 后 使 用 “cd /datal ”命令 切换 到 “datal ”目录 ， 以 下 的 操作 将 会 在 “datal” 目 录 
中 完成 。 首 先 使 用 yum 升级 或 安装 系统 工具 类 库 〈 需 要 系统 接 入 互联 网 )。 


[root@~]# yum -y install gcc gcc-c+t+ make wget autoconf libjpeg libjpeg-devel 


libpng libpng-devel freetype freetype-devel libxml2 libxml2-devel zlib zlib-devel 

glibc glibc-devel glib2 glib2-devel bzip2 bzip2-devel ncurses ncurses-devel curl 

curl-devel ssse2fsprogs e2fsprogs-devel krb5 krb5-devel libidn libidn-devel openssl 

openssl-devel openldap openldap-devel nss_ldap openldap-clients openldap-servers 

接着 就 可 以 安装 Nginx 了 ， 这 里 安装 的 版 本 为 1.3.8 Nginx DI URL 处 理 模 块 基于 pcre 
实现 ， 所 以 在 安装 Nginx 前 需要 安装 pere 类 库 。 读 者 可 以 在 官方 网 站 中 获取 到 相关 源 代 码 
包 ， 也 可 以 使 用 以 下 命令 进行 获取 。 


[root@~]# wget http://soft.beauty-soft.net/lib/nginx-1.3.8.tar.gz 


[root@~]# wget http://soft.beauty-soft.net/lib/pcre-8.10.tar.gz 
下 载 完成 后 ， 束 可 安装 pere 了 。 

[root@~]# tar zxvf pcre-8.10.tar.gz 

[root@~]# cd pcre-8.10/ 

[root@~]# ./configure 

[root@~]# make && make install 


[root@~]# cd ../ 
然后 创建 Nginx 运行 用 户 及 用 户 组 ， 这 里 将 使 用 WWW 用 户 及 用 户 组 运行 Nginx。 
12 E E 


第 1 章 开发 前 准备 


[root@~]# groupadd -g 402 www 

[root@~]# useradd www -g www 

接着 就 可 以 安装 Nginx 1.3.8 了 。 

[root@~]# tar zxvf nginx-1.3.8.tar.gz 

[root@~]# cd nginx-1.3.8 

[root@~]# ./configure --user=www --group=www --prefix=/usr/local/nginx --with- 

http_stub_status_module --with-http_ssl_module 

[root@~]# make 

[root@~]# make install 

通过 前 面 的 步 又，Nginx 的 安装 就 完成 了 。 使 用 ls /usr/local/nginx 命令 可 以 查看 Nginx 
安装 后 的 目录 结构 。Nginx 非常 灵活 ， 默 认 情 况 下 并 不 需要 做 任何 配置 即 可 正音 运行 。 


[root@~]# /usr/local/nginx/sbin/nginx 


[root@~]# pstree |grep nginx 
| -nginx---nginx 

HJJ, HV HRSA TE P 即 可 看 到 Nginx 成 功 提 示 页 面 。 需 要 注意 的 是 ， 
此 时 的 Nginx 只 能 用 于 做 代理 服务 器 、 绥 存 服务 吉 以 及 解释 静态 文件 。 接 下 来 将 使 用 源 代码 
TRJ IAZ PHP 5.3.6, LEE Nginx 能 够 解释 PHP 程序 。 

2. SS PHP 

DU DUU ll Nginx 本 号 不 文 持 解释 cgi 程序 ， 所 要 必须 借助 于 第 三 方 FastCGI EMAK 
现 动态 程序 解释 。 对 于 PHP WA, HH FastCGI 管理 右 有 spawn-fcgi 及 php-fpm。 其 中 
spawn-fcgi 是 lighttp 服务 器 的 一 部 分 ， 现 在 已 经 成 为 一 个 独立 的 项 目 ， 在 安装 时 可 以 以 独立 
的 安 状 包 进 行 安 装 ; php-fpm 在 PHP 5.3.2 版 本 之 前 也 是 一 个 独立 的 项 目 ， 现 在 已 经 成 为 
PHP 的 一 部 分 ， 并 以 插件 的 方式 整合 到 PHP 中 。 接 下 来 将 以 PHP 5.3.6 为 例 ， 详 细 介 绍 PHP 
的 安装 过 程 。 

(1) 安装 PHP 依赖 库 

在 安装 PHP 5.3.6 之 前 ， 需 要 安装 或 更 新 PHP 依赖 库 。 这 里 将 继续 以 源 代 码 包 安装 方式 
进行 安装 ， 首 先 下 载 源 代码 包 。 


[root@~]# cd /datal 


[root@~]# wget http://soft.beauty-soft.net/lib/libiconv/libiconv-1.13.1.tar.gz 
[root@~]# wget http://soft.beauty-soft.net/lib/merypt/libmerypt-2.5.8.tar.gz 
[root@~]# wget http://soft.beauty-soft.net/lib/merypt/merypt-2.6.8.tar.gz 
[root@~]# wget http://soft.beauty-soft.net/lib/mhash/mhash-0.9.9.9.tar.gz 
[root@~]# wget http://soft.beauty-soft.net/lib/libiconv/php-5.3.6.tar.gz 
PRERA. H PRR T 

[root@~]# tar zxvf libiconv-1.13.1.tar.gz 

[root@~]# cd libiconv-1.13.1/ 

[root@~]#./configure --prefix=/usr/local 

[root@~]# make 


[root@~]# make install 


D WE 
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[Foots] F cd css 

[root@~]# tar zxvf libmcrypt-2.5.8.tar.gz 

[root@~]# cd libmerypt-2.5.8/ 

[root@~]#./configure 

[root@~]# make 

[root@~]# make install 

[root@~]# /sbin/ldconfig 

[root@~]# cd libltdly/ 

[root@~]#./configure --enable-ltdl-install 

[root@~]# make 

[root@~]# make install 

[rootte E CA soy ney 

[root@~]# tar zxvf mhash-0.9.9.9.tar.gz 

[root@~]# cd mhash-0.9.9.9/ 

[root@~]#./configure 

[root@~]# make 

[root@~]# make install 

[root@~]# cd se 

[root@~]# ln -s /usr/local/lib/libmcrypt.la /usr/lib/libmerypt.la 

[root@~]# ln -s /usr/local/lib/libmcrypt.so /usr/lib/libmerypt.so 

[root@~]# ln -s /usr/local/lib/libmcrypt.so.4 /usr/lib/libmcrypt.so.4 

[root@~]# ln -s /usr/local/lib/libmcrypt.so.4.4.8 /usr/lib/libmcrypt.so.4.4.8 

[root@~]# ln -s /usr/local/lib/libmhash.a /usr/lib/libmhash.a 

[root@~]# ln -s /usr/local/lib/libmhash.la /usr/lib/libmhash.la 

[root@~]# ln -s /usr/local/lib/libmhash.so /usr/lib/libmhash.so 

[root@~]# ln -s /usr/local/lib/libmhash.so.2 /usr/lib/libmhash.so.2 

[root@~]# ln -s /usr/local/lib/libmhash.so.2.0.1 /usr/lib/libmhash.so.2.0.1 

[root@~]# ln -s /usr/local/bin/libmerypt-config /usr/bin/lipbmcrypt-config 

[root@~]# tar zxvf mcrypt-2.6.8.tar.gz 

[root@~]# cd mcrypt-2.6.8/ 

[root@~]# /sbin/ldconfig 

[root@~]# ./configure 

[root@~]# make 

[root@~]# make install 

[root@~]# cd sau 

(2) %3 PHP 

安装 PHP 相对 比较 简单 ， 只 需要 在 编 详 时 加 入 php-fpm 扩展 即 可 。 此 外 为 了 能 够 连接 
MySQL 数据 库 ， 还 需要 加 入 相应 的 扩展 。 
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[root@~]# tar zxvf php-5.3.6.tar.gz 


[root@~]# cd php-5.3.6 


[root@~]# ./configure —--prefix=/usr/local/php --with-config-file- 
path=/usr/local/php/etc --with-iconv-dir --with-freetype-dir --with-jpeg-dir -- 
with-png-dir --with-zlib --with-libxml-dir=/usr --enable-xml --disable-rpathħh 一 一 
enable-discard-path --enable-magic-quotes --enable-safe-mode --enable-bcmath -- 
enable-shmop  --enable-sysvsem  --enable-inline-optimization  —--with-curl 一 一 WILL 七 有 一 
curlwrappers --enable-mbregex --enable-fastcgi --enable-fpm --enable-force-cgi- 


redirect --enable-mbstring --with-mcrypt --enable-ftp --with-gd --enable-gd-native- 

ttf --with-openssl --with-mhash --enable-pcntl --enable-sockets --with-xmlrpc -—- 

enable-zip --enable-soap --without-pear --with-gettext --with-mime-magic --with- 

mysql=mysqlnd --with-mysqli=mysqlnd --with-pdo-mysql=mysqlnd 

[root@~]# make ZEND_EXTRA_LIBS='-liconv' 

[root@~]# make install 

通过 前 面 的 步骤 ，PHP 就 安装 完成 了 ， 接 下 来 只 需要 创建 php-fpm 配置 文件 即 可 启动 
PHP 引擎 。 

(3) 配置 文件 

PHP 5.3.6 安装 完成 后 ， 并 没有 日 动 生成 php-fpm 配置 文件 ， 但 提供 了 一 个 php- 
fpm.conf.default 模板 文件 ， 接 下 来 需要 将 该 文件 复制 一 份 ， 用 于 正式 的 配置 文件 。 


[root@~]# cp /usr/local/php/etc/php-fpm.conf.default cp /usr/local/php/etc/php- 


fpm.conf 

打开 php-fpm.conf 配置 ， 这 里 需要 修改 运行 用 户 及 用 户 组 〈 与 Nginx 相同 )。 此 外 还 需 
要 修改 pm.start_servers 〈 动 态 方 式 下 起 始 进 程 数 量 ) 及 pm.min_spare_servers (动态 方式 下 最 
小 php-fpm 进程 数量 ) 配置 项 。 如 以 下 代码 所 示 。 

[global] 


pid = /usr/local/php/var/run/php-fpm.pid 
error_log = /usr/local/php/var/log/php-fpm.log 


log_level = notice 


[www | 

listen = /tmp/php-cgi.sock 
user = www 

group = www 

listen = 127,0.0,13 9000 


pm = dynamic 


pm.max_children = 20 
pm.start_servers = 20 
pm.min_spare_servers = 10 


listen 配置 项 表示 监听 地 址 及 端口 ，Nginx ZA AREIA PHP, MWER H R RE 
URL 请 求 到 该 监听 地 址 。 

(4) 局 动 php-fpm 

PHP 5.3.6 安装 完成 后 ， 并 没有 自动 生成 php.ini 配置 文件 ， 但 在 源 代码 包 中 提供 了 一 个 
php.ini-production 模板 文件 ， 接 下 来 需要 将 该 文件 复制 一 份 ， 用 于 正式 的 配置 文件 。 
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[root@~]# cp /opt/sdal/sft/php-5.3.6 /php.ini-production /usr/local/php/etc/php.ini 

通过 六 面 的 步骤 ，php-fpm 及 PHP 已 经 安装 完成 了 。 最 后 只 需要 局 动 php-fpm 即 可 。 

[root@~]# /usr/local/php/sbin/php-fpm 

php-fpm 主 进程 默认 使 用 9000 应 口 ， 要 检测 是 否 成 功 运 行 ， 可 以 使 用 netstat -nlpt |grep 
9000 命令 查看 。 

3. BÆ Nginx 

前 面 提 到 过 ， 虽 然 Nginx DATKI, (HEITI Nginx 与 PHP ZRK, ENZ 
间 是 两 个 不 同 的 服务 。 接 下 来 将 通过 配置 Neinx， 实 现 解 秋 PHP。 首 先 打 开 Nginx 配置 
人 

[root@~]# vi /usr/local/nginx/conf/nginx.conf 

然后 修改 运行 用 户 名 及 用 户 组 为 www， 并 且 在 server APIA PHP 请 求 转 发 文 持 ， 
如 以 下 代码 所 示 。 


location ~ n pap es 28 
{ 


#fastcgi_pass unix:/tmp/php-cgi.sock; 
rasce pass 2 1390005 
fastcgi_index index.php; 

Tielvoe TECGL CONLE 


} 
如 上 述 代 人 码 所 示 ，fastcgi_pass 配置 项 表示 fastcgi 管理 需 地 址 ， 即 php-fpm DDT. include 


表示 包含 外 部 文件 ， 该 文件 用 于 配置 fastcgi 运行 方式 ， 内 容 如 以 下 代码 所 示 。 


fastcgi_param GATENWAY INTERFACE CGI/1.1; 
fastcgi_param SERVER_SOFTWARE nginx/$nginx version; 


fastcgi param QUERY STRING Şquery_string; 
fastcgi_param REQUEST_METHOD Şrequest_method; 
fastcgi_param CONTENT_TYPE Şcontent_type; 
fastcgi_param CONTENT_LENGTH Şcontent_length; 


fastcgi_param SCRIPT_FILENAME sdocument_ roots$fastcgi script name; 


fastcgi param SCRIPT_NAME SEaASECOL Seriot Neame; 
fastcgi_param REQUEST_URI GEET Dl 
fastcgi_param DOCUMENT_URI Şdocument_uri; 
fastcgi_param DOCUMENT_ROOT Sdocument root: 


fastcgi_param SERVER_PROTOCOL Şserver_protocol; 


fastcgi_param REMOTE_ADDR Şremote_addr; 
fastcgi_param REMOTE PORT sremote port,; 
fastcgi param SERVER ADDR $sserver addr; 
fastcgi param SERVER PORT Şserver_port; 
fastcgi_param SERVER_NAME $server_name; 


# PHP only, required if PHP was built with --enable-force-cgi-redirect 
fastcgi_param REDIRECT_ STATUS 200; 


fcgi.conf 文件 路 径 需 要 与 nginx.conf 文件 平 级 ， 默 认 情 况 下 该 文件 并 不 存在 ， 需 要 开发 
人 员 手 动 创建 。 


[root@~]# touch /usr/local/nginx/conf/fcgi.conf 


最 终 ，nginx.conf 配置 文件 代码 如 下 。 
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user WWW WNWW7 


worker_processes l1; 


error_log logs/error.log; 
#error_log logs/error.log notice; 
#error_log logs/error.log info; 


pid logs/nginx.pid; 


events { 
use epoll: 


worker_connections 1024; 


Mego 
include mime .types; 
default_type application/octet-stream; 
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#log_format main 'S$remote addr - S$remote user [$time locall] "$request" ' 


# '$status $body_bytes_sent "$http_referer" 


# WSImeeD user agent WOMeeD EE 


#access_log logs/access.log main; 


sendfile on; 
#tcp_nopush on; 
#keepalive_timeout 0; 
keepalive_timeout 65; 


#gzip on; 
server { 
listen 80; 
server_name localhost; 
root /home/wwwroot; 
#charset koi8-r; 
#access_log logs/host.access.log main; 
location / 
root /home/wwwroot; 
index index.html index.htm index.php; 
} 
error_page 404 /404.html; 


# redirect server error pages to the static page 
# 
error_page 500 502 503 504 /50x.html; 
location = /50x.html { 
root html; 


/50x.html 


# proxy the PHP scripts to Apache listening on 127.0.0.1:80 


# 
#location ~ \.php$ { 
# prozy pass heceos//127:-0-0.15 
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+} 


# pass the PHP scripts to FastCGI server listening on 127.0.0.1:9000 
# 
error_page 405 =200 Q405; 
location Q405 
{ 
root /home/wwwroot; 
} 
location ~ .*\. (php |php5)?$ 
{ 
#fastcgi_pass unix:/tmp/php-cgi.sock; 
fastcgi_pass 127.0.0.1:9000; 
fastcgi_index index.php; 


include fcgi.conf; 


# deny access to .htaccess files, if Apache's document root 
# concurs with nginx's one 

# 

#location ~ /人 .ht { 

# deny all; 

zl 


# another virtual host using mix of IP-, name-, and port-based configuration 


# 


#server { 


# 


} 


= 


listen 8000; 
listen somename:8080; 


server_name somename alias another, alias: 


location / { 
root html; 


index index.html index.htm; 


HTTPS server 


#server { 


# 
# 
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listen AAS a 


server_name localhost; 

ssl on; 
ssl_certificate cert .pem; 
ssl Certificate key Cert. keye 


ssl_session_timeout 5m; 


ssl_protocols SSLv2 SSLv3 TLSv1; 
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ssl_ciphers HIGH: IaNULL: !MD5; 
ssl_prefer_server_ciphers on; 
location / { 

root html; 


index index.html index.htm; 


Sa ao ae au ua u 


| 


标 粗 的 即 为 需要 改动 的 配置 ， 完 成 后 只 需要 重 局 Nginx 即 可 。 

[root@~]# pkill nginx 

[root@~]# /usr/local/nginx/sbin/nginx 

需要 注意 的 是 ， 如 果 php-fpm PE ARTELE. mi HIE EHS Dr, Ukim J 
司 9000 mO, REKA Kk. A ol El Ehome/wwwroot 创建 PHP 文件 ， 以 便 测试 PHP 
是 否 运 行 正常 。 

4. 安装 MySQL 

MySQL 的 安装 方式 有 很 多 种 ， 其 中 最 常用 的 有 使 用 yum 和 rpm 安装 ， 这 里 继续 以 源 代 
码 的 安装 方式 安装 MySQL 5.5.3。 安 装 步骤 如 下 。 

[root@~]# cd /datal 


[root@~]# wget http://soft.beauty-soft.net/lib/mysql-5.5.3-m3.tar.gz 

[root@~]# groupadd mysql 

[root@~]# useradd -g mysql mysql 

[root@~]# tar zxvf mysql-5.5.3-m3.tar.gz 

[root@~]# cd mysql-5.5.3-m3/ 

[root@~]#./configure --prefix=/usr/local/mysql/ --enable-assembler --with-extra- 
charsets=complex --enable-thread-safe-client --with-big-tables --with-readline --with-ssl 
--with-embedded-server --enable-local-infile --with-plugins=partition, innobase, myisammrg 

[root@~]# make 

[root@~]# make install 

需要 注意 的 是 在 编译 时 务必 开局 partition 插件 ， 该 插件 用 于 实现 数据 表 分 区 ， 本 书 第 
7 革 7.5 节 将 介绍 数据 表 分 区 的 内 容 。 安 状 耗 时 根据 机 右 性 能 有 所 出 入 ， 一 般 在 30 一 90min 
PAE 

安装 完成 后 ， 在 /usrlocalmysql 目录 中 可 以 找到 MySQL 相关 可 执行 文件 ， 同 时 在 
/etc 目录 下 可 以 找到 my.cnf 配置 文件 。 需 要 注意 的 是 /etc/my.cnf 配置 文件 并 不 能 满足 局 动 
MySQL 的 环境 需求 ， 事 实 上 安装 程序 已 经 在 /usr/local/mysql/share/mysql/ 目 录 中 创建 了 多 
SI MySQL 配置 文件 模板 ， 这 里 将 使 用 my-medium.cnf 模板 文件 作为 配置 文件 ， 将 其 罕 换 
my.cnf 文件 即 可 。 

[root@~]# cp /usr/local/mysql/share/mysql/my-medium.cnf /etc/my.cnf 

my-medium.cnf RO AFA m EEE A CA REEE MySQL 的 需要 。MySQL 
安装 完成 后 ， 所 有 可 执行 工具 存放 于 /usr/local/mysgl/bin/ 目 录 。 其 中 mysql_install_db 是 一 个 
用 初始 化 MySQL 数据 库 的 工具 ， 在 局 动 MySQL 前 需要 使 用 该 工具 创建 MySQL 默认 数据 
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库 及 存放 目录 【安装 程序 并 不 会 日 动 创建 )。 
[root@~]#/usr/local/mysql/bin/mysgql_install db --basedir=/usr/local/mysql --datadir=/ 
usr/local/mysql/var --user=mysql 
[root@~]# chown www:www -R /usr/local/mysql/var 
执行 完 mysql_install_db 命令 后 ， 问 导 将 会 在 /usr/local/mysql/var 创建 名 为 “mysql” 的 数 
据 库 ， 该 数据 库存 放 看 与 用 户 授 权 、 系 统 运 行 状态 等 关键 数据 。 通 过 前面 的 步 又 ， 接 下 束 可 
以 启动 MySQL 主 程序 了 。 
[root@~]# nohup /usr/local/mysql/bin/mysqld_safe --user=mysql 
前 面 虽然 创建 了 “mysql” 数 据 库 ， 但 并 没有 创建 登录 用 户 数 据 。 接 下 来 将 使 用 
mysqldump TRAJES KHF Rn rei, 
[root@~]# /usr/local/mysql/bin/mysqladmin -u root password root 
PIE, MySQL HZC T o PLEH pstree |grep mysql 命令 检测 mysqld_safe EFE 
序 是 否 存在 ， 或 者 直接 在 终端 登录 MySQL 数据 库 。 
[root@~]# /usr/local/mysql/bin/mysql -uroot -proot 
为 了 便于 操作 ， 读 者 可 以 使 用 phpmyadmin Ze" H H I A ET E. RA 
MySQL 时 下 接 结 束 mysqld 主 进 程 即 可 。 


[root@~]# pkill mysqld 


1.3 ”开发 工具 介绍 


要 进行 PHP 开发 ， 并 不 需要 特定 的 开发 工具 ， 使 用 Windows 上 自 带 的 记事 本 程序 就 可 
以 开发 Web 应 用 ， 由 于 PHP 是 一 门 脚 本 语言 ， 所 以 并 不 像 C# 或 Java 那样 需要 使 用 IDE 进 
行 调试 、 编 译 等 。 要 运行 PHP， 则 先 需 要 将 该 文件 放置 到 Web 服务 嚣 下， 然后 输入 文件 
URL 即 可 。 但 是 随 看 PHP 越 来 越 智 能 化 ， 功 能 模块 越 来 越 多 ，Web 应 用 变 得 更 加 复杂 ， 
特别 是 后 期 软件 的 测试 、 团 队 合作 等 都 需要 像 Visual Studios Eclipse 那样 的 集成 化 开发 环 
境 。PHP 是 一 门 成 熟 的 Web 技术 开发 语言 ， 经 过 了 几 波 Web 技术 风潮 后 ， 专 为 PHP 而 设 
计 的 集成 化 开发 环境 已 经 非常 多 ， 有 些 是 商业 的 ， 有 些 则 是 免费 和 开源 的 ， 本 节 将 会 介绍 
几 套 比较 流行 的 PHP 集成 化 开发 环境 ， 开 发 人 员 使 用 这 些 工 具 能 够 明显 地 提高 开发 速度 并 
减少 程序 的 错误 。 


1.3.1 PHP Coder 


PHP Coder 是 一 球 比 较 著 名 的 PHP 集成 开发 环境 ， 它 的 界面 非常 友好 ， 能 够 为 PHP FF 
发 提供 集成 化 的 开发 、 调 试 、 代 人 码 感知 、HTML 高 亮 显示 、 项 目 管理 等 完善 的 功能 。PHP 
Coder 的 调试 功能 非常 强大 ， 在 没有 架设 PHP 服务 喜 的 环境 上 ， 也 能 够 完成 应 用 程序 的 调试 
及 开发 。PHP Coder 是 免费 并 且 开 源 的， 能 够 在 Windows 和 Linux 操作 系统 下 使 用 ， 开 发 人 
员 无 论 在 Windows 还 是 Linux 下 ， 所 和 耐 对 的 DE 界面 都 相差 不 大 ， 这 点 是 非 笛 友好 的 ; 另 
一 个 比较 值得 称道 的 是 PHP Coder 的 代码 感知 功能 是 非常 强大 和 快速 的 ， 这 一 点 对 开 友 Web 
应 用 是 非常 实用 的 。PHP Coder 目前 没有 中 文 版 ， 界 和 面 如 图 1-11 所 示 。 
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图 1-11 PHP Code 界面 


1.3.2 PHP Editor 


PHP Editor 全 称 为 DzSoft PHP og 是 DzSoft SAMFARA 208 H Urgë e pi 
PHP 开发 工具 ，PHP Editor 体积 非常 小 ， 是 一 球 轻 量 级 但 注音 实效 的 IDE。 能 够 提供 代码 跟 
踪 、CSS 智能 提示 、PHP 代码 调试 、PHP 函数 自动 完成 等 实用 功能 ，DzSoft PHP Editor 目前 
最 新 版 本 为 5.4， 狐 版 本 提供 了 ZendFramework 等 框架 的 开发 支持 ，DzSoft PHP Editor 界面 
非常 简洁 ， 如 图 1-12 所 示 。 
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图 1-12 DzSoft PHP Editor 主 界面 


1.3.3 NetBeans IDE 
NetBeans IDE 出 目 于 大 名 易 易 的 Sun 公司 ，NetBeans IDE 并 不 是 专业 的 PHP 开发 工 
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具 ， 而 是 专业 级 的 Java 开发 工具 。 但 由 于 Sun 公司 对 PHP 技术 的 重视 ， 在 5.0 以 后 的 版 本 
中 提供 了 强大 的 PHP 开发 支持 。NetBeans IDE 与 另 一 个 重量 级 的 Java 开发 工具 Eclipse 非常 
类 似 ， 它 们 都 能 够 提供 从 开发 到 程序 友 布 等 一 系列 企业 级 开发 广 持 ， 这 两 僚 工 具 与 微软 的 
Visual Studio 各 有 于 秋 ， 是 最 流行 的 开发 工具 。 使 用 NetBeans IDE 开发 PHP， 无 论 是 从 效率 
还 是 程序 的 稳定 性 来 讲 NetBeans IDE 都 是 理想 的 PHP 开发 工具 。NetBeans IDE 遵行 GPL 发 
行 协 议 ， 大 多 数 情 况 下 个 人 或 公司 都 可 以 免费 使 用 ;， NetBeans IDE 能 够 完美 地 运行 在 
Windows, Linux, MacOS 等 操作 系统 上 ， 界 面 如 图 1-13 所 示 。 
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图 1-13 NetBeans IDE 界面 


1.3.4 Eclipse PDT 


Eclipse 是 IBM 的 一 个 商业 项 目 ，2001 年 11 月 起 IBM 将 Eclipse 贡献 给 开源 社区 ， 从 
Ir GC T Eclipse 的 地 位 。Eclipse 是 Java 开发 人 员 的 首选 开发 工具 ， 见 悉 Java 的 谈 者 应 议 
都 对 这 一 工具 非 第 熟悉 。Eclipse 通过 PDT 插件 来 提供 PHP 开发 文 持 ，Eclipse PDT (PHP 
Development Tool Kit) 能 够 让 编写 PHP 变 得 向 单 和 高 效 ， 下 面 将 介绍 Eclipse PDT 的 安装 
和 使 用 。 

Eclipse PDT 的 安装 有 两 种 方式 ， 一 种 是 和 直接 通过 Eclipse 的 扩展 进行 安装 ; 为 一 种 
是 下 载 带 PDT 插件 的 Eclipse 。 下 面 将 以 Eclipse 图 Wekome 
3.5.2 作为 基础 ， 讲 解 Eclipse PDT 插件 的 安装 

首先 确保 计算 机 上 已 经 下 载 了 Eclipse， 并 已 
经 配置 好 了 相关 JDK 和 JAR 运行 环境 ， 双 击 
Eclipse 局 动 应 用 程序 ， 在 Eclipse 主 环境 中 依次 
单 击 束 单 栏 上 的 “Help” 一 “Install New Software...”， 
如 图 1-14 所 示 。 图 1-14 选择 Install New Software, .命令 


在 弹出 的 对 话 框 “Work with” 一 栏 中 输入 PDT 下 载 地 址 http://downloads.zend.com/pdt， 
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完成 后 下 方 的 列表 框 中 将 会 列 出 可 用 的 插件 ， 如 图 1-15 所 示 。 
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图 1-15 添加 PDT 插件 


将 所 需要 的 插件 选中 ， 然 后 单 击 “Next” 按 钮 ， 进 入 插件 下 载 对 话 框 ， 单 击 “Finish ” 
按钮 即 可 。 


如 果 下 载 市 PDT 插件 的 Eclipse， 只 需要 双击 Eclipse 局 动 程序 即 可 ， 此 时 Eclipse Wite 


供 了 完美 的 PHP 编程 文 持 。 带 PDT 的 Eclipse 下 载 网 址 为 http://www.eclipse.org/pdt/ 
downloads/， 主 界面 如 岁 1-16 所 示 。 
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图 1-16 Eclipse PDT 主 界面 
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1.3.5 Zend Studio 


Zend Studio 是 Zend Technologies 所 开发 的 一 和 套 商 业 化 PHP 开 友 工具 ， 近 年 来 层 获 大 
奖 ， 被 众多 企业 广 为 采 用 。 尽 管 是 商业 化 软件 ， 但 由 于 Zend Technologies 对 PHP 的 深耕 细 
作 ，Zend Studio 能 够 为 PHP 开发 帝 来 无 与 伦比 的 编程 体验 。Zend Studio 文 持 众多 企业 级 开 
发 ， 并 且 能 够 与 Zend Technologies 其 他 的 一 些 重量 级 产品 进行 无 颖 整合 ， 如 Zend Guard, 
ZendFramework、Zend Platform 等 。Zend Studio 在 6.0 之 前 采用 独立 安 闭 包 进 行 发 行 。6.0 及 
以 后 的 版 本 则 基于 Eclipse PDT 发 行 了 全 新 的 商业 化 套件 ， 目 前 最 新 的 厂 本 为 9.0。 下 面 分 别 
对 经 典 的 Zend Studio 5.5 和 成 熟 的 Zend Studio 8.0 进行 讲解 ， 方 便 读 者 选择 。 

1. Zend Studio 5.5 

Zend Studio 5.5 是 Zend Studio 采用 独立 安 产 包 发 行 版 本 的 最 后 一 个 版 本 ， 也 是 非常 经 典 
的 一 个 版 本 ， 目 前 仍 被 大 量 的 PH 开发 者 和 企业 所 采用 。Zend Studio 5.5 使 用 Java 语言 开发 
而 成 ， 能 够 对 PHP 提供 完善 的 支持 ， 这 种 商业 化 的 支持 与 其 他 的 PHP 开发 工具 相 比 ， 显 得 
更 加 深入 和 全 面 。 比 如 Zend Studio 5.5 能 够 提供 CYS、SVN 等 团队 开发 文 持 ， 也 能 够 文 持 
WSDL、 数 据 库 可 视 化 、ZendFramework 等 企业 级 开发 。Zend Studio 5.5 内 置 完 善 的 调试 机 
Hh HAWARA -ANEA PHP 解释 器 ， 开 有 人 员 不 需要 额外 Web 服务 右 即 可 完成 简单 
的 PHP 调试 ， 甚 到 不 需要 将 文件 保存 也 能 进行 PHP 的 调试 ， 这 为 调试 PHP UU ée CEO T 
非常 便捷 的 手段 。 

Zend Studio 5.5 PE J EHRM He OH Ce, DEEN Eclipse 那样 对 项 目的 文件 组 织 、 结 
构 进 行 细 化 及 分 类 ，Zend Studio 5.5 项 目 管理 器 能 够 根据 文件 的 增 减 ， 实 时 地 更 狐 项 目 信 
县 ， 开 发 人 员 可 以 通过 项 目 管 理 窗口 了 解 当 前 项 目的 结构 状态 。 

Zend Studio 5.5 代码 编辑 器 的 效率 非常 局， 能 够 对 PHP Du, mër, fe, 
量 、 类 实例 、 方 法 每 进行 全 方位 文 持 (如 代码 提示 、 代 人 码 或 注释 格式 化 、 实 时 错误 检查 
等 )， 还 能 够 对 HTML, JavaScripts XML 等 语言 提供 友好 的 文 持 。Zend Studio 5.5 相对 
Eclipse 来 说 比较 小 巧 ， 但 启动 速度 却 有 点 慢 ， 界 面 如 图 1-17 ra, 
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图 1-17 Zend Studio 5.5 主 界面 
如 图 1-16 所 示 ，Zend Studio 5.5 ER MARKSA AIA 5 部 分 ， 如 下 所 示 。 
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> A: 项 目 管 理 面 板 。 该 面板 是 Zend Studio 5.5 的 项 目 管理 窗口 ， 开 发 人 员 可 以 在 该 面 
板 中 对 项 目 文件 进行 操作 ， 如 添加 、 删 除 、 修 改 等 。 
> B: 代码 编辑 器 。 代 码 编辑 器 是 Zend Studio 中 最 重要 的 面板 ， 也 是 集成 化 开发 环境 
的 精华 毛 在， 判断 一 个 DE 是 人 盏 强大 在 一 定 程 上 度 上 取决 于 代码 编辑 器 ，Zend Studio 
5.5 的 代码 编辑 器 对 PHP 的 支持 一 直 为 业界 称道 ， 能 够 提供 完善 的 PHP 开发 支持 。 
> C: 调试 输出 面板 。 该 面板 相当 于 一 个 小 型 服务 器 ， 可 以 实时 地 浏览 程序 的 运行 结 
果 ， 如 HTML、Txt 等 。 
> D: Zend Platform 事件 管理 面板 。 该 面板 提供 Zend Platform 错误 输出 、 调 试 跟踪 
等 信息 管理 。 
> E: 调试 信息 管理 面板 。 访 面板 输出 调试 时 程序 的 健康 状态 等 信息 。 
2. Zend Studio 8.0 
Zend Studio 8.0 是 基于 Eclipse PDT 发 行 的 一 个 全 狐 的 商业 套件 ， 能 够 提供 企业 级 的 
PHP 开发 。Zend Studio 8.0 基于 Eclipse 的 强大 项 目 管 理 功 能 ， 整 合 了 Zend 目 家 的 多 项 业 
务 ， 提 供 了 全 新 的 PHP 调试 引擎 、 代 码 感 知 、 智 能 分 析 等 一 系列 强大 功能 。Zend Studio 8.0 
能 够 深度 文 持 MVC 框架 开发 ， 如 MVC 的 关联 模块 的 创建 、 模 型 数据 库 可 视 化 管理 等 。 
Zend Studio 8.0 支持 多 国语 言 ， 如 图 1-18 所 示 即 为 中 文 版 的 Zend Studio 8.0 主 界面 。 
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图 1-18 Zend Studio 8.0 主 界面 


如 图 中 所 示 ，Zend Studio 8.0 无 论 是 从 界面 布局 还 是 菜单 位 置 都 与 Eclipse 非 第 类 似 ， 习 
惯 于 使 用 Eclipse 的 读者 应 该 非常 容易 上 手 。Zend Studio 8.0 是 Zend 推出 的 商业 套件 ， 开 发 者 
或 企业 必须 要 购买 商业 许可 才能 进行 使 用 。 当 然 也 可 以 下 载 试 用 片 ，Zend 提供 了 标准 版 和 企 
业 版 供 开 发 者 免费 试用 ， 下 载 地 址 为 http://www.zend.com/en/products/studio/downloads。 


1.3.6 Adobe Dreamweaver 
严格 来 讲 Dreamweaver 并 不 是 PHP 开发 工具 ， 之 所 以 放 到 本 节 来 讲 ， 是 因为 Dreamweaver 
E E 25 


PHP MVC 开发 实战 


能 够 提供 一 流 的 网 页 编程 体验 ， 对 PHP 来 说 更 重要 的 是 编写 PHP 中 的 HTML 代码 部 分 。 
Dreamweaver 是 业界 最 赣 名 的 网 页 议 计 工具 ， 它 能 够 提供 可 视 化 网 页 放 计 功能 ， 是 网 页 开发 从 
业 人 员 不 可 缺少 的 工具 之 一 。Dreamweaver 不 仅 能 够 编写 可 视 化 的 HIML， 借 助 于 其 完善 的 代 
人 码 编写 带 ， 还 能 够 编号 编码 正确 ， 羔 容 性 好 的 动态 编程 语言 ， 如 PHP、JSP 等 。 

Dreamweaver 不同 于 专业 性 强 的 IDE, Dreamweaver 定位 于 可 视 化 网 页 设计 ， 能 够 无 颖 
融合 Adobe 的 其 他 重量 级 套件 ， 提 供出 色 的 网 页 设计 功能 。 作 为 PH 开发 人 员 ， 特 别 是 
MVC 设计 人 员 ， 主 要 使 用 Dreamweaver 来 设计 视图 显示 层 ， 配 合 专 业 的 PHP IDE 能 够 为 编 
程 市 来 无 与 伦比 的 效率 。 

Dreamweaver 虽然 提供 了 可 视 化 的 PHP 开发 功能 ， 但 是 文 持 的 深度 比较 简单 及 原始 。 由 
于 Dreamweaver 对 HTML, CSS, JavaScript (Jquery, ExtJs) 的 友好 文 持 ， 所 以 在 混合 编程 
时 显得 非常 有 效率 ， 另 外 Dreamweaver 的 项 目 管理 器 非 第 全 面 和 强大 ， 不 仅 提供 了 出 色 的 文 
件 组 织 功 能 ， 还 提供 FTP 远程 管理 ，SVN 版 本 库 控 制 等 实用 功能 。 

Dreamweaver 界面 非常 友好 ， 能 够 让 初次 接触 Wb 设计 的 开发 人 员 迅 速 上 手 。 
Dreamweaver 还 提供 了 众多 的 快捷 键 ， 能 够 显 彰 提高 开发 人 员 的 效率 。 如 图 1-19 所 示 ， 显 
示 了 Dreamweaver CS3.0 的 界面 概况 。 
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图 1-19 Dreamweaver CS3.0 主 界面 


如 图 中 所 示 ，Dreamweaver 主 界面 主要 分 为 6 个 部 分 ， 分 别 如 下 。 

> A: 功能 亲 音 区。 主要 提供 了 Dreamweaver 的 工具 、 控 件 、 可 视 化 表单 等 。 

> B: RABS Ato Dreamweaver 的 代码 编写 器 非 常 高 效 ， 在 编号 HTML 时 可 以 在 代码 
模式 与 可 视 化 模式 之 间 进 行 切换 。 

> C: 属性 面板 。Dreamweaver 的 属性 面板 非常 实用 ， 特 别 是 在 设计 可 视 化 的 HTML 时 
就 显得 非常 高 效 。 属 性 面板 会 根据 HTML 元 素 的 属性 而 出 现 相 应 的 属性 选项 。 

> D: 信息 区 。 该 区域 可 以 显示 Dreamweaver 编程 时 的 异常 信息 、 校 验 信 息 等 。 

> E: CSS 管理 面板 。 该 面板 提供 了 对 CSS 的 可 视 化 管理 ， 开 发 人 员 可 以 使 用 该 面板 
完成 CSS 的 创建 、 修 改 、 删 除 等 。 
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> F: 项 目 文件 管理 区 。 该 面板 提供 了 强大 的 项 目 管理 功能 ， 能 够 提供 FTP 文件 上 传 、 
文件 新 旧 对 比 等 实用 的 功能 。 
Dreamweaver 在 Adobe 公司 的 大 力 推 三 下 ， 目 前 已 经 成 为 最 主流 的 网 页 设计 工具 ， 最 狐 
版 本 为 Dreamweaver CS 5.5。 新 厂 本 提供 了 对 众多 新 技术 的 文 持 ， 如 HIMLS、 多 屏幕 协 
同 、CSS 3.0、 移 动 开发 、Jquery 等 。 


1.3.7 VS.PHP 


VS.PHP HI Visual Studio for PHP, "CR JCX 公司 (微软 的 合作 伙伴 〉 针 对 微软 Visual 
Studio 套件 而 开发 的 一 个 商业 插件 ， 基 于 Visual Studio 对 项 目的 友好 支持 ， 提 供 一 站 式 的 
PHP 编程 。 如 末 使 用 过 Visual Studio 的 程序 员 会 感觉 VS.PHP 非常 亲切 和 熟悉 ，VS.PHP wt 
是 为 这 样 的 一 群 人 而 设计 的 。 

使 用 VS.PHP 开发 应 用 程序 ， 能 够 像 C# 那 样 流 畅 、 高 效 地 编写 代码 ， 程 序 员 能 够 在 一 
站 式 的 环境 里 专注 PHP 应 用 的 实现 ， 而 不 需要 在 各 个 工具 间 来 回 切换 。Visual Studio A 
历经 时 代 洗 礼 的 成 熟 开 发 人 套件， 能 够 提供 从 测试 环境 到 生产 环境 的 全 程 文 持 ，VS.PHP 基于 
Visual Studio 的 民 好 操作 性 ， 提 供 了 出 色 的 编程 能 力 和 语言 代 人 码 管理 能 力 ， 是 专业 PHP FR 
人 员 需 要 了 解 的 工具 之 一 。 

VS.PHP 的 安装 非常 容易 ， 官 方 提供 了 Web 在 线 安 装 包 和 MSI 离线 安装 包 。VS.PHP 是 
商业 产品 ， 个 人 或 企业 使 用 首先 需要 购买 商业 许可 ， 当 然 也 可 以 下 载 30 天 试用 版 ， 下 载 地 
址 为 http://www.jcxsoftware.com/jcx/vsphp/downloads 。 安 闭 完 成 后 可 以 在 Visual Studio 中 的 
“New Project” 对 话 框 中 找到 “PHP Projects ”模板 ， 如 图 1-20 所 示 。 


图 1-20 Visual Studio for PHP 
值得 注意 的 是 ，VS.PHP 必须 要 根据 Visual Studio RANA a, SE 


致 安装 成 功 后 却 找 不 到 项 目 模 板 的 情况 。 


O mo: 在 安装 VS.PHP 前 ， 必 须要 确保 已 经 成 功 安装 了 Visual Studio， 否 则 VS.PHP 将 会 退出 安装 ;如 
果 需 要 完整 的 PHP 调试 环境 ， 还 需要 安装 XDebug。 
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1.4 SVN 版 本 控制 


如 条 只 是 一 个 人 在 开发 应 用 程序 ， 那 么 只 需要 将 文件 保存 到 一 个 指定 的 目录 即 可 ， 如 采 
担心 数据 丢失 ， 只 需要 将 整个 目录 备份 ; 但 是 如 末 是 一 个 团队 〈3 人 以 上 ) 合作 项 目 ， 就 需 
要 进行 版 本 控制 和 备份 了 。SVN 是 一 套 版 本 控制 系统 ， 能 够 在 多 人 开发 项 目 时 提供 数据 备 
份 、 冲 突 处 理 、 新 旧 检 测 等 功能 ， 近 年 来 已 经 被 众多 企业 所 采用 ， 接 下 来 将 深入 介绍 SVN 
的 理论 知识 与 实际 应 用 。 


1.4.1 SVN 介绍 


SVN 全 称 为 SubVersion， 它 是 一 套 版 本 控制 系统 ，SVN 近年 来 大 受 欢迎 的 一 个 重要 原 
因 是 SVN 解决 了 团队 合作 项 目 中 的 许多 问题 。 在 运用 版 本 控制 之 前 ， 一 个 团队 的 开发 人 员 
通常 使 用 共享 文件 夹 的 方式 进行 协同 编程 ， 有 些 甚至 使 用 QQ 等 文件 传输 工具 进行 相互 传 
递 ， 在 极 少 人 参与 的 项 目 环境 中 不 会 成 为 问题 ， 但 在 一 些 比 较 大 的 团队 项 目 ， 使 用 上 述 方式 
进行 团队 协作 ， 效 率 是 非常 低 的 ， 而 且 整 个 项 目 是 不 可 控 的 ， 一 旦 需要 找 回 旧 的 文件 ， 这 时 
就 是 一 项 复杂 的 任务 。SVN 是 一 套 版 本 控制 系统 ， 简 单 地 说 就 是 一 套 自动 备份 系统 。 这 个 比 
喻 虽然 不 太 恰 当 ， 但 使 用 SVN 完全 不 用 担心 数据 的 丢失 ， 只 要 从 一 开始 就 使 用 SYN， 哪 怕 
项 目 快 结束 了 ， 也 能 找到 刚 开 始 时 的 文件 内 容 。 

事实 上 SVN 远 不 止 备 份 那么 简单 ， 之 所 以 这 样 比 喻 ， 是 为 了 让 没 接触 过 SVN 的 读者 有 
个 直观 印象 ， 方 便 接 下 来 的 学 习 。SVN 使 用 数据 库 形 式 保存 文件 版 本 ， 项 目 中 的 文件 上 只 需要 
有 一 个 文件 发 生 了 改变 ， 项 目 版 本 号 就 会 自动 加 1， 版 本 号 是 唯一 能 够 让 文件 回 滚 的 凭证 ， 
有 了 版 本 号 开发 人 员 束 可 以 大 刀 阔 借 地 进行 程序 开发 了 ， 所 以 有 些 程序 员 杀 切 地 把 版 本 控制 
称 为 “项 目 开发 的 时 光 机 器 ”。SVN 版 本 控制 流程 如 图 1-21 所 示 。 


eg 
SVN 
repos 
= p 
EH E cn 
源码 . z 源码 
程序 员 I 程序 员 2 程序 员 3 


R| 1-21 SVN 流程 图 


28 DO NO 


第 1 章 开发 前 准备 


图 1-21 简单 地 演示 了 SVN 多 人 协同 开发 时 的 流程 。 首 先 SVN 服务 占 为 项 目 创 建 版 本 
库 ， 当 “程序 员 1” 用 update 命令 将 数据 同步 〈 当 本 地 项 目 内 容 为 空 时 需要 使 用 checkout 命 
令 进 行 同步 ) 到 SVN 数据 库 时 ，SVN 服务 器 将 与 本 地 计算 机 进行 数据 同步 ， 完 成 后 即 可 处 
于 开发 状态 ;此 时 “程序 员 1” 上 所 编写 的 程序 将 只 能 目 己 看 到 ， 当 “程序 员 1” 确 定 所 编写 
的 程序 完成 时 ， 可 以 “commit( 提 交 )” 到 SVN 服务 器 ;“ 程 序 员 2” 与 “程序 员 3” 可 以 使 
用 “update” 命 令 获 取 被 改动 的 文件 ，SVN 版 本 控制 系统 内 置 有 一 套 高 效 的 版 本 冲突 处 理 机 
制 ， 防 止 3 个 程序 员 同 写 一 个 文件 时 造成 的 冲突 。 

SVN 是 一 僚 成 熟 的 版 本 控制 系统 ， 经 过 多 个 版 本 的 演进 ，SVN 已 经 能 够 运行 在 
Windows, Linux, MacOS 等 操作 系统 上 ， 并 且 提 供 了 商业 化 文 持 ，SVN 之 前 的 版 本 控制 系 
统 为 CVS (Concurrent Versions System), CVS 是 专 为 C、C++ 设 计 的 ， 由 于 其 先天 性 的 缺陷 
《如 CSV 只 能 记录 单一 文件 版 本 ，CYVS 安全 策略 单一 等 ) 所 以 并 没有 得 到 广泛 的 应 用 。SVN 
JET CVS 的 特点 ， 改 进 了 CVS 的 版 本 控制 方式 ， 使 用 了 类 似 于 数据 库 的 方式 进行 版 本 省 
FE; SVN 一 直 强 调用 户 目 行 管理 的 特点 ， 所 以 强化 了 SVN 服务 器 的 安全 功能 ， 使 得 项 目 开 
发 者 对 各 目 所 管理 的 模块 更 加 清晰， 易于 控制 。SVN 的 完善 、 易 用 已 经 成 为 了 团队 项 目 合 作 
的 代名词 ， 作 为 一 个 PHP 程序 员 ， 熟 悉 SVN 已 经 成 为 一 门 必要 的 知识 ， 本 节 将 会 详细 讲解 
SVN HREH 


1.4.2 SVN 的 安装 


SVN TARE mA m RS mA mA ARE Windows, Linux 等 操 
ERRE, Sal Windows 使 用 得 比较 多 ， 接 下 来 将 详细 介绍 在 Windows 平台 安装 SVN 客 
户 闹 及 服务 占 病 的 全 过 程 。 

1. Windows SVN 服务 端的 安装 

d) 安装 服务 包 

接 下 来 将 以 SubVersion 1.6.5 为 基础 ， 讲 解 SVN 的 安装 过 程 。 要 安装 SVN 首先 需要 下 载 
相应 的 安装 包 ，SubVersion 的 安装 包 是 免费 的 ， 谈 者 可 以 在 国内 外 下 载 网 站 下 载 到 ， 或 者 登录 
官方 网 站 http:/www.open.collab.neUcn/downloads/subversion/ 进 行 下 载 。 把 下 载 的 压缩 包 解 压 
后 ， 找 到 Setup-Subversion-1.6.5.msi LF, NIZ, ma, WB 1-22 所 示 。 

Dein) “Next?” RE, Eier H K, AE D:subversion 作为 安装 目录 ， 
如 图 1-23 所 示 。 


ren | 期 Subversion Setup 
Welcome to the Subversion Setup | Chk Next to instal to the defaut folder or dd Change to dhore mas 
Wizard | 


Ths wl stl Dheera LA Sen on er Cem, 


Wi Per ed ia ec dor al ciho 本 bone 
before cmefrsge, 


Cid best bo contiat er Canad to cii Saup, 


图 1-22 SubVersion 安装 问 导 图 1-23 ”指定 SubVersion 的 安装 目录 
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根据 问 导 提示 ， 完 成 SubVersion 的 安装 〈 这 个 过 程 需要 重新 局 动 计算 机 )， 安 装 完 成 
后 进入 Di subversion 但 看 目录 下 的 文件 ， 该 目录 下 的 bin 目录 即 为 SubVersion 的 核心 服务 
包 文 件 。 

(2) 创建 版 本 库 

正如 前 面 所 言 ，SVN 之 所 以 能 够 进行 版 本 控制 必须 依赖 于 SVN 的 版 本 数据 库 。 在 使 用 
SVN 服务 之 前 ， 首 先 要 创建 版 本 库 〈Repository)， 创 建 步骤 如 下 。 

首先 在 D 盘 下 创建 一 个 空 的 目录 ， 并 命名 为 svn， 该 目录 用 于 存放 SVN 版 本 库 ; 打开 
命令 行 工 具 ， 切 换 到 SubVersion 的 核心 文件 目录 ; 然后 使 用 svnadmin create di:\svn\reposl GI 
建 版 本 数据 库 。 当 命令 成 功 运 行 后 可 以 进入 Di\svn 目录 ， 此 时 可 以 看 到 svn 目录 下 已 经 多 出 
了 reposl 文件 夹 ， 该 文件 夹 即 为 SVN 的 版 本 控制 库 ， 进 入 reposl 文件 来， 可 以 看 到 
SubVersion 管理 终端 已 经 生成 了 夏 本 控制 库 所 需要 的 目录 与 文件 ， 如 网 1-24 Bro, 


图 1-24 SubVersion 版 本 库 目 录 结 构 


(3) 局 动 SubVersion 宿主 服务 

经 过 前 面 的 步骤 ，SVN 的 厂 本 库 已 经 成 功 建立 ， 要 让 客户 山 访 问 SVN 的 版 本 库 ， 需 要 
启动 SubVersion Die "CIR. OT SVN 的 版 本 库 默 认 情 况 下 只 允许 匿名 用 户 以 只 读 的 方式 
进行 访问 ， 这 里 为 了 方便 演示 ， 将 创建 一 个 名 为 test, WN test 的 用 户 ， 访 用户 能 够 以 读 
和 写 的 方式 对 reposl 版 本 库 进 行 访问 。 要 实现 使 用 用 户 名 和 密码 访问 ， 需 要 对 reposl 版 本 
库 的 配置 文件 作 一 些 简单 的 修改 ， 其 体 步骤 如 下 。 

首先 进入 版 本 库 所 在 的 目录 Di\svn\reposl\conf， 打 开 该 目录 下 的 svnserve.conf 文件 ， 访 
文件 即 为 SVN 版 本 库 配 置 文 件 ， 找 到 “[general] ”市 点 ， 然 后 将 “# password-db = passwd” 
中 的 注释 符 去 反 〈 即 删除 “#” 符 号 ，# 号 后 不 能 接 空 格 )， 这 样 reposl 版 本 库 束 能 够 以 用 户 
名 与 密码 的 形式 进行 访问 了 。 

接 下 来 需要 创建 用 户 名 和 密码 ， 打 开 有 目录 下 的 passwd 文件 (可 以 使 用 记事 本 打开 )， 该 
文件 即 为 版 本 库 的 用 户 配置 文件 ， 在 “[users]” 节 点 后 添加 test=test， 即 添加 test 用户， 该 用 
FRIN test， 最 终 passwd 文件 代码 如 下 所 示 。 
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### This file is an example password file for svnserve. 
### Its format is similar to that of svnserve.conf. As shown in the 
### example below it contains one section labelled [users]. 


### The name and password for each user follow, one account per line. 


[users] 
# harry = harryssecret 


# sally = sallyssecret 
CESET =CESE 


ALATA, MN S RAE EMAIT EH svnserve.exe -daemon MS, JA 
动 SubVersion Ze ERS., mot LEH TortoiseSVN 等 SVN ZS P imne E Hd o T o 

(4) 将 SubVersion 服务 加 入 操作 系统 服务 

WACAKA J SubVersion 服务 ， 此 时 如 果 需 要 访问 必须 确保 命令 行 工 具 处 于 激活 
状态 ， 一 旦 关闭 命令 行 终端 ，SubVersion 服务 进程 将 会 结束 。 这 里 可 以 使 用 Windows 的 sc 
命令 将 SubVersion 牡 主 服 务 加 入 操作 系统 中 ， 这 样 在 系统 局 动 时 将 会 目 动 局 动 SubVersion 服 
务 ， 命 令 如 下 。 


sc create SvnService binpath= " d:\subversion\bin\svnserve.exe --service -—-root 


D:\svn" depend= "TCPIP" start= auto 

2. Ster 

DR e mi ec hl LC, DREA mta SubVersion 的 管理 工具 ，SubVersion 内 置 了 一 
个 客户 端 管理 工具 svn.exe， 该 工具 只 能 在 命令 行 下 使 用 。 下 面 将 介绍 另 一 天 比较 流行 的 
SubVersion 客户 疹 管 理工 具 TortoiseSVN。 

TortoiseSVN 能 够 紧密 地 集成 到 Windows 右键 快捷 沫 单 中 ， 提 供出 色 的 操作 体验 ， 使 用 
TortoiseSVN 能 够 明显 地 提高 工作 效率 。TortoiseSVN 是 免费 开源 的 ， 同 样 能 够 运行 在 Windows 
及 Linux 等 主流 操作 系统 上 ， 接 下 来 将 详细 讲解 在 Windows 操作 系统 上 安装 TortoiseSVN. 

要 安装 TortoiseSVN ， 首 先 需 要 下 载 相 应 的 安 冯 包 ， 下 载 网 址 为 http://tortoisesvn.net/ 
downloads.html。 TortoiseSVN WIZ] Aa, AA “Next” AAEE. ZRT 
JKodt me km T br ee, MBMA 1-25 PR, HUH TortoiseSVN 安装 成 功 了 。 


图 1-25 Tortoises VN 快捷 菜单 
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1.4.3 TortoiseSVN 的 简单 使 用 


前 面 介 绍 了 TortoiseSVN 的 安装 ， 接 下 来 将 讲解 TortoiseSVN 的 使 用 。TortoiseSVN 的 使 
用 非 党 直观、 友好 ， 读 者 只 需要 理解 以 下 儿 个 重要 的 TortoiseSVN 概念 ， 残 可 以 像 操 作 
Windows 资源 管理 需 一样 进行 版 本 控制 。 

> 通信 协议 : TortoiseSVN 是 一 个 客户 端 ， 用 于 连接 到 SubVersion 服务 器 ， 并 操作 服务 

Zë DIR AR, CET: svn:// 通信 协议 ， 与 常见 的 FTP 客户 姗 使 用 ftp:/ 通 信 协 议 
不 一 样 ， 这 也 就 意味 看 只 要 是 基于 svn:/ 通 信 协 议 的 服务 ， 都 能 够 使 用 TortoiseSVN 
进行 操作 。 
> 本 地 版 本 库 : 一 个 用 户 在 参与 项 目 时 ， 通 常 是 没有 修改 项 目 文件 权限 的 ， 一 旦 分 配 
了 合法 的 SVN 用 户 名 及 密码 ， 就 意味 看 该 用 户 是 一 个 合法 的 项 目 参 与 者 ， 此 时 该 用 
户 只 有 将 SubVersion 服务 器 上 的 项 目 文件 下 载 到 本 地 ， 才 能 进行 同步 编程 。 

> 文件 上 传 : 用 户 在 本 地 改动 了 项 目 中 的 文件 ， 必 须要 将 文件 上 传 到 SubVersion 服务 
髓 ， 才 能 和 被 其 他 用 户 奏 看 到 ， 同 时 版 本 库 会 目 动 增 1。 

> 版 本 号 : SubVersion 的 版 本 写 并 不 是 针对 单一 文件 的 ， 而 是 针对 整个 项 目 ， 这 也 束 意 
味 看 束 算 项 目 中 的 某 个 文件 只 作 和 人 简单 的 修改 ， 一旦 进行 提交 操作 ， 整 个 项 目的 版 本 
写 永 会 日 增 1。 

1. TortoiseSVN 的 菜单 

下 面 将 通过 实践 的 方式 ， 消 化 上 述 理论 化 的 知识 。 在 讲解 TortoiseSVN 的 实际 应 用 前 ， 

首先 简单 介绍 TortoiseSVN JLA HE. 

> Svn Checkout: 该 命令 用 于 项 目 参 与 者 第 一 次 下 载 SubVersion 版 本 库 文件 。 
TrotoiseSVN 一 Repo-browser: 浏览 SubVersion WRA EM H AR. 
TrotoiseSVN 一 Export: 导出 版 本 库 中 的 项 目 文 件 。 

TrotoiseSYN 一 Create repository here: 创建 版 本 库 ， 访 命令 的 作用 与 svnadmin create 
命令 相同 。 

> TrotoiseSVN>Import: 将 现 有 的 项 目 文件 导入 SubVersion WW Am, 

> TrotoiseSYN 一 Settings: 该 命令 能 够 弹出 TortoiseSVN 评 细 设置 窗口 。 

2. 使 用 步骤 

下 面 通过 一 个 示例 讲解 TortoiseSVN 实际 应 用 。 这 里 继续 使 用 前 面 创建 的 reposl 版 本 库 
作为 项 目 版 本 库 ， 项 目的 本 地 路 径 为 D:\php\ProjectDemol。 首 先 在 项 目 中 创建 一 个 
index.php 文件 ， 然 后 使 用 TortoiseSVN 人 快捷 键 ， 依 次 选择 “TrotoiseSVN ” 一 “Import” 命 
令 ， 将 ProjectDemol 目录 导入 reposl 版 本 库 中 ， 如 图 1-26 所 示 。 

单 击 “OK” 按钮 ， 将 会 要 求 输入 用 户 名 和 密码 (这 里 的 账号 及 密码 即 为 前 面 创建 的 
SVN 登录 用 户 名 及 密码 ， 即 test 账号 )。 完 成 后 ，ProjectDemol 项 目 将 会 被 导入 reposl 版 本 
库 中 ， 如 网 1-27 所 示 ， 如 果 导 入 失败 ， 首 先 检查 SVN 答 主 服务 是 人 否 已 经 局 动 ， 然 后 检查 用 
尸 名 与 密 公 是 否 正 硼 。 

此 时 可 以 通过 “Repo-browser” 命 令 租 看 reposl 版 本 库 中 的 文件 。 此 时 但 看 reposl 版 
本 库 的 实际 存放 路 径 ， 除 了 看 到 版 本 库 体 积 增 大 外 ， 其 他 的 目录 与 文件 结构 一 点 也 没 发 生 
改写。 
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er ProjectDemol - Tortoises WN Import.. Finished! 


Achion Path Mime type 
Ading D: iphp'frojectDemo Lindex. php 
Competed Atrevison: 7 


图 1-26 Import 对 话 框 图 1-27 导入 成 功 对 话 框 


初次 接触 SVN 的 读者 经 党 认为 将 项 目 导入 版 本 库 之 后 ， 就 可 以 进行 编程 了 。 事 实 
上 ， 前 面 所 做 的 项 目 导 入 只 是 创建 版 本 库 源 文件 而 已 ， 读 者 需要 前 先 明 白 ， 项 目 源 文件 不 
是 一 个 人 在 开发 的 ， 而 是 多 人 一 起 开发 的 。 当 前 登录 的 用 户 为 test， o; 
必须 要 将 版 本 库 中 的 源 文件 取出 ， 所 以 应 该 使 用 “Svn Checkout” 命 令 获取 源 文件 ， 
如 下 。 
首先 在 D:\php\ 下 创建 一 个 目录 〔 不 能 使 用 中 文 名 称 )， 并 命名 为 test， 然 后 进入 test H 
录 ， 按 下 鼠标 右键 ， 在 弹出 TortoiseSVN 快捷 菜单 中 选择 “Svn Checkout” 命 令 ， 如 图 1-28 
所 示 。 
ZC “URL of repository” 一 栏 中 输入 “repos1” 版 本 库 svn 地 址 ， 单 击 “OK ”按钮 ， 此 
时 TortoiseSVN 将 会 开始 下 载 版 本 库 中 的 源 文 件 ， 如 图 1-29 所 示 。 


Path 

D'vdeveztVp be eteg fb igl Ema ty chs modis comnpis count worda. php 
D: php iestiip bbs opp ib iii Smarty hemeohbercoemgger Cat php 

D: php iest wn bag app Ub Wa sm ty 和 加 请 En himi Paie php 


D: php hiest iip _bbsapp ilib Wa Emar tu ks modis regex eben php 
D: 'php\iesiiip_bbs app ib Wima ty phagns function. counter php 
O'edeeztn bc areg Gg Ganze ty plagne ec peu him options. php 
D'eéevestn ae app ilib Smarty phgins modis oompie. Tom charset, php 
D:iphpliestiip_bbslappihbWiSmatyiplgns uh ex pon hn seiect free php 

EI | 


图 1-28 ”下载 厂 本 库 源 文件 图 1-29 Svn Checkout 对 话 框图 


当 版 本 库 源 文件 下 载 完 毕 后 ， 进 入 test 目录 ， 可 以 看 到 版 本 库 中 的 目录 与 文件 都 被 下 载 
到 本 地 了 ， 此 时 才 正 式 处 于 团队 项 目 开 发 的 状态 。 可 以 看 到 test 目录 下 的 文件 都 被 
TortoiseSVN 目 动 加 上 了 文件 图 标 ， 因 为 此 时 还 没 对 文件 进行 更 改 ， 项 目的 版 本 号 〈 即 项 目 
没有 文件 被 改动 过 ) 与 版 本 库 中 的 项 目 版 本 号 是 一 致 曲 ， 所 以 文件 图 标 都 显示 为 已 经 更 新 状 
No TortoiseSVN 常见 状态 图 标 如 图 1-30 所 示 。 

TortoiseSVN 币 见 状态 网 标 含义 如 下 。 

> normal: 向 规 状 态 ， 此 状态 下 的 文件 夹 《〈 文 件 ) 厂 本 号 与 版 本 库 中 保存 一 致 。 

> ignored: 忽略 ， 该 文件 夹 (文件 ) 已 经 被 忽略 ， 提 交 时 被 排除 。 
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VW WW WM Y 


> 


ei Ca A È 


normal ignored conflicted locked 
readonly added non-versioned modifie 
deleted 


图 1-30 TortoiseSVN 常见 状态 图 标 


confilcted: 冲突 ， 该 文件 夹 (文件 ) 已 产生 了 冲突 ， 需 要 做 冲突 处 理 。 

locked: 锁定 ， 访 文件 夹 〈 文 件 ) 已 经 被 上 锁 ， 操 作 前 需要 解锁 。 

readonly: Kik, ALR UF) 只 能 谈 取 。 

added: 添加 观察 名 单 ， 该 文件 夹 〈 文 件 ) 已 经 被 添加 到 观察 名 单 ， 能 被 SVN 识别 。 
non-versioned: 无 版 本 控制 ， 该 文件 夹 〈 文 件 ) 还 没有 被 SVN 识别 。 

modified: 已 修改 ， 该 文件 夹 〈 文 件 ) 已 经 被 修改 ， 但 还 没 与 版 本 库 进 行 同步 。 
deleted: 己 删 除 ， 访 文件 夹 (文件 ) 已 经 被 其 他 用 户 删除 。 


中 index.php 文件 并 弹出 鼠标 右键 ， 在 弹出 的 TortoiseSVN 快捷 菜单 中 选择 “SVN Commit” 
命令 ， 将 此 次 修改 保存 到 SVN 版 本 库 ， 如 图 1-31 所 示 。 


[F] Select | deselect all 


Zeen lods 


图 1-31 SVN Commit 对 话 框 


在 “Recent messages” 一 栏 中 输入 本 次 修改 的 目 定义 信息 ， 然 后 按 下 “OK” 按 钮 ， 这 


样 项 目 参 与 者 束 能 够 及 时 地 但 看 到 被 修改 过 的 文件 了 ; 如 末日 己 需 要 僵 看 项 目的 最 新 状态 ， 
可 以 使 用 “SVN Update” 命 令 ， 将 本 地 文件 更 独到 与 SubVersion WAE 29. 
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前 面 介 绍 了 TortoiseSVN 的 文件 下 载 、 同 步 、 上 传 等 简单 功能 ， 这 些 功 能 是 
TortoiseSVN 最 基础 和 最 各 见 的 功能 ， 接 下 来 将 讲解 TortoiseSVN 的 高 级 使 用 方法 ， 这 些 功 能 


主要 包含 文件 删除 、 历 史 版 本 、 新 旧 对 比 、 冲 突 机 制 等 ， 


数 实际 应 用 开发 中 的 需要 。 
1. 文件 删除 


笠 握 这 些 功 能 的 使 用 ， 可 以 请 足 多 


当 版 本 库 与 本 地 项 目 同步 后 ， 要 删除 项 目 中 的 茶 个 文件 ， 初 学 者 往往 直接 选中 文件 进行 删 


除 ， 这 样 一 来 再 次 使 用 “SVN Update” 时 ， 发 现 文件 突然 
又 回来 了 。 正 确 的 做 法 是 使 用 TortoiseSVN 操作 采 单 中 的 
“Delete ”命令 (在 项 目 文件 上 依次 选择 “TrotoiseSVN” 
一 “ Delete”), WWB] 1-32 所 示 。 命 令 执 行 后 需要 再 选 
择 “SVN Update” 命 令 ， 以 便 其 他 程序 员 看 到 项 目 发 生 
BE 

2. 历史 版 本 

所 谓 的 历史 版 本 也 就 是 SVN 的 文件 找 回 功能 。 假 
WEH T “Delete” MEMBR S F, EET UR a e 
次 找 回 该 文件 ， 历 史 版 本 功能 就 派 上 用 场 了 。SVN 是 以 
版 本 写作 为 版 本 控制 依据 的 ， 只 要 项 目 中 有 一 个 文件 发 
生 了 更 改 ， 并 已 经 录入 版 本 库 ， 那 么 SVN 服务 器 将 把 
项 目 版 本 控制 提升 1 "RA, amer Lade 
要 找 回 旧 的 文件 ， 只 需要 输入 相应 的 版 本 号 即 可 。 在 项 


Eu Show log 
员 Repo-browser 

Do, Check for modifications 
E Revision graph 

€ Update to revision- 

vd Rename... 


I Release lock 
T Branch/tag.- 


Pé Switch... 

Y Merge.. 

GL Blame.. 

E Create patch... 
bn Properties 

? Help 

ÑE Settings 

ee About 


图 1-32 ”TrotoiseSVN 文件 操作 荣 单 


目的 空白 区 域 上 依次 选择 “TrotoiseSVYVN” 一 “Update to revision.… ”命令 ， 在 弹出 的 
“Update ”对话 框 中 输入 需要 回 深 的 版 本 写 即 可 ， 如 图 1-33 所 示 。 
如 果 对 版 本 号 不 是 很 清楚 ， 可 以 点 击 “Show log” 按 钮 ， 在 弹出 的 更 新 日 志 对 话 框 中 选 


拌 相应 的 记录 ， 如 图 1-34 rz 


图 1-33 Update 对话 框 


图 1-34 更 新 日 忘 对 话 框 
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3. 新 旧 对 比 

所 谓 的 新 旧 对 比 是 TrotoiseSVN 中 一 项 实用 的 功能 ， 它 能 够 让 程序 员 记 得 哪些 代码 是 日 
己 写 的 。 假 设 有 2 名 程序 员 ， 分 别 使 用 A 与 B RR, X 2 名 程序 员 共 同 对 一 个 文件 进行 编 
o A EFREN B 程序 员 继 续 对 该 文件 进行 编辑 ， 此 时 吏 可 以 通过 “Diftt with 
previous version” 命 令 奏 看 文件 被 A 程序 员 修 改 的 部 分 ， 这 就 是 新 旧 对 比 的 典型 应 用 。“Diff 
with previous version ”对话 框 如 图 1-35 所 示 。 


OS ,txt : Working Base 
-=1 Ho ` 


93999939 
For Help, press FL Serall horizontally with Ctrl-Serolhwheel Left View: -1 Right | 


图 1-35 Diff with previous version 对 话 框 


如 图 1-35 所 示 ， 左 边 部 分 表示 A 程序 员 删 除 的 内 容 ， 右 边 部 分 表示 B 程序 员 新 增 的 内 
容 ， 通 过 左右 图 对 比 ， 可 以 清楚 地 了 解 文件 被 更 改 状 况 。 

d. 冲突 机 制 

TrotoiseSVN 文件 版 本 冲突 机 制 能 够 处 理 两 名 程序 员 同 时 对 同一 个 文件 进行 编程 时 所 产 
生 的 冲突 。SVN 是 以 版 本 号 作为 唯一 版 本 控制 依据 的 ， 默 认 情 况 下 低 版 本 的 项 目 文件 不 能 后 
局 版 本 的 项 目 文 件 进行 履 盖 提交 ， 所 以 SVN 也 提供 了 版 本 冲突 处 理 机 制 ( 类 似 于 操作 系统 
WFE). EH A 和 B 程序 员 都 将 项 目 版 本 号 更 新 到 了 10 号 版 本 (最 狐 版 本 )， 并 且 他 
们 同时 对 同一 个 文件 进行 编程 ， 当 A 程序 员 提 交 修 改 后 的 文件 时 项 目 版 本 与 将 会 变 为 11; 
此 时 B 程序 员 的 文件 还 处 于 10 号 版 本 ， 当 B 提交 文件 时 ， 由 于 版 本 库 的 项 目 版 本 号 为 11, 
根据 低 厂 本 号 不 能 上 履 辣 高 版 本 号 的 原则 ， 这 时 训 突 惑 会 产生 了 ， 如 疼 1-36 rar, 


er OS tGet = TortoiseSVN Commit.. Finished! 


| Path 
D:'php'user 2. tet 


图 1-36 ”文件 提交 冲突 
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为 此 ，TrotoiseSVN 提供 了 “Edit conficts” 命 令 用 
于 处 理 上 述 问 题 。 一 旦 冲突 发 生 ， 首 先 使 用 “SVN 
Update” 命 令 与 版 本 库 进 行 同步 ， 以 保证 版 本 号 一 致 ， 
此 时 可 以 发 现 目录 下 多 出 了 几 个 候选 文件 ， 如 图 1-37 
所 示 。 

冲突 候选 文件 根据 版 本 库 的 版 本 写 不 同 而 出 现 不 同 
的 文件 名 ， 这 里 将 以 上 述 4 个 文件 为 例 ， 介 绍 它们 的 具 。 erie indexphp. 
体 含义 ， 分 别 如 下 。 

> index.php: 冲突 产生 的 文件 ， 所 有 产生 冲突 的 SE ee 

文件 将 以 黄色 感叹 号 为 标识 。 打 开 该 文件 将 会 看 到 产生 冲突 的 部 分 内 容 。 

> index.php.mine: 本 地 项 目 原 文件 。 

> index.php.r3: 版 本 号 为 3 的 index.php 文件 。 

> index.php.r4: 版 本 号 为 4 的 index.php 文件 。 

选中 冲突 文件 〈index.php)， 依 次 选择 “TrotoiseSVN ” 一 “Edit conficts”( 编 辑 冲 突 )” 
命令 ， 在 弹出 的 “编辑 冲突 ”对 话 框 中 清楚 地 显示 了 服务 占 版 本 与 本 地 项 目 版 本 的 冲突 片段 
内 容 ， 如 图 1-38 所 示 。 


| index.php. 
mine 


index.php 


ZS index.php.mine - Tortoseherde 
File Edit Navigate View Help 


"MG TIAJ vede yT 


| Theirs - index.php.r4 * 


el 1111111111114 = |1 1111111111114 z 
-2 222222222222 小 2 33333333333 


For Help, press EL. Scroll horizontally with Ctri-Serollwheel 


图 1-38 编辑 冲突 对 话 框 


如 图 中 所 示 ，A 区 显示 的 是 服务 器 上 的 版 本 文件 ，B 区 显示 的 是 本 地 文件 被 改动 过 的 文 
件 ; C 区 显示 两 者 合并 后 的 结果 。 产 生 冲 突 的 内 容 区 域 使 用 遇 融 (红色 ) 标识 。 

既然 知道 了 冲突 的 文件 内 容 ， 那 么 就 可 以 进行 冲突 处 理 了 。 解 决 冲突 的 方式 在 该 对 话 框 
中 变 得 非常 容易 ， 步 又 如 下 。 

这 里 假设 需要 将 本 地 内 容 攻 盖 厂 本 库 中 的 内 容 ， 只 需要 在 B 区 的 编辑 框 中 选中 高 之 鸣 内 
容 区 域 ， 然 后 按 下 鼠标 右键 ， 选 择 “Use this text block” ME, WKI 1-39 所 示 。 
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Use this text block 
Use thishole file 
Use text block from 'mine' before ‘theirs' 
Use text block from here" before "mune! 


Copy 


图 1-39 re nn 


XERE EHH A bh A E A GC SVN 服务 器 中 的 内 容 如 果 需 要 将 远程 SVN 的 
AARNA, Uc A 区 编辑 框 中 选择 相同 的 操作 即 可 ; ERR Ié Uu e CG UH 
Im), WMR ek OI ër LFE E, LEE EES “Use this whole file” 命 令 即 可 ; 最 后 按 下 
《CtrltS〉 快 捷 键 ,你 存 更改 ， 并 关闭 编辑 冲突 对 话 框 。 

上 述 操 作 完 成 后 ， 还 需要 做 最 后 的 一 步 操 作 ， 选 中 币 贡 色 感 叹 写 的 index.php 文件 ， 
依次 选择 “TrotoiseSVN” 一 “ Resolved.. CHER)” ME, 确认 冲突 已 经 解决 。 在 冲突 
解决 完成 后 ， 目 录 下 的 候选 文件 将 会 日 动 消失 ， 产 生 冲 突 的 index.php 文件 已 经 变更 为 可 
提交 状态 。 

通过 TrotoiseSVN 的 冲突 解决 机 制 ， 可 以 看 到 SVN 对 文件 的 冲突 处 理 提 供 了 非 单 友好 
的 操作 ， 整 个 流程 非常 共有 弹性 ， 在 实际 应 用 开发 中 可 以 与 项 目的 参与 者 商量 后 决定 使 用 哪 
个 冲突 文件 。 


1.4.5 SVN 版 本 库 权 限 配 置 


SVN 捉 供 了 完善 的 用 户 管 理 权 限 ， 用 户 管理 权限 在 多 人 项 目 开 发 中 具有 非 癌 重要 的 作 
用 ， 它 能 够 明确 地 让 开发 人 员 在 目 己 的 职 贡 范围 内 编写 代码 。 例 如 技术 经 理应 该 能 够 控制 整 
个 项 目的 文件 恋 写 ， 而 程序 员 应 该 根据 职责 分 配 相应 的 权限 ， 例 如 做 前 合 界面 设计 的 不 应 该 
能 够 编写 后 台 PHP 模块 文件 。SVN 使 用 authz 文件 作为 权限 配置 文件 ， 该 文件 提供 了 用 户 组 
功能 ， 组 别 权限 配置 能 够 具体 到 项 目的 文件 夹 。 下 面 通过 一 个 简单 的 示例 介绍 authz 权限 文 
件 的 配置 。 

假设 当前 项 目 有 3 名 程序 员 共 同 参与 ， 匿 名 用 户 没 有 任何 权限 《默认 情况 下 有 读 的 权 
限 )， 只 需要 在 passwd 文件 中 配置 3 名 用 户 即 可 ， 代 人 码 如 下 。 


### This file is an example password file for svnserve. 


### Its format is similar to that of svnserve.conf. As shown in the 
### example below it contains one section labelled [users]. 
### The name and password for each user follow, one account per line. 


[users] 

# harry = harryssecret 
# sally = sallyssecret 
ceiba=12345 

ES@SESESSE 
manager=12345 


如 上 述 代码 所 示 ，ceiba 与 test 用 户 都 是 普通 的 程序 员 ; manger 是 项 目 负责 人 。 在 项 目 
"D äs It bbs 和 user 目录 ， 分 别 代 表 论 坛 系统 与 会 员 中 心 ; 为 了 便于 分 工 合作 ， 这 里 将 分 配 
bbs 目录 给 用 户 ceiba， 让 其 具有 可 读 可 写 的 权限 ; user 目录 分 配给 test， 让 其 具有 可 读 可 写 
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的 权限 。 用 户 ceiba 进入 user 目录 时 ， 限 制 其 只 能 浏览 但 禁止 提交 文件 ，test 用 户 访问 bbs 
目录 时 也 限制 其 只 能 浏览 但 禁止 提交 文件 。 

根据 上 述 权限 分 配 要 求 ， 只 需要 在 authz 创建 一 个 权限 组 admin， 这 个 权限 组 只 赋 给 项 
目 负责 人 manger: 然后 再 分 别 对 项 目下 的 其 他 目录 单独 配置 权限 ，authz 文件 代码 如 下 。 


### This file is an example authorization file for svnserve. 

### Its format is identical to that of mod_authz_svn authorization 

### files. 

### As shown below each section defines authorizations for the path and 
### (optional) repository specified by the section name. 

### The authorizations follow. An authorization line can refer to: 


### 一 a single user, 

### 一 a group of users defined in a special [groups] section, 

### - an alias defined in a special [aliases] section, 

### all authenticated users, using the '$authenticated' token, 

### 一 only anonymous users, using the '$anonymous' token, 

### 一 anyone, using the '*' wildcard. 

ZEÄ 

### A match can be inverted by prefixing the rule with '~'. Rules can 

### grant read ('r') access, read-write ('rw') access, or no access 

EE 

[aliases] 

# joe = /C=XZ/ST=Dessert/L=Snake City/O=Snake OLL, Ltd./OU=Research 
Institute/CN=Joe Average 

[groups] 


admin = manager 
# harr and sally = harry,sally 
# harry_sally_and joe = harry,sally,&joe 


[/] 


Qadmin = rw 


[/bbs] 
ceiba = rw 
test = r 
[/user] 
test = rw 


ceiba = r 


[/foo/bar] 
harry = rw 
&joe = r 


db 


[repository:/baz/fuz] 
Qharry_and sally = rw 


Eeer a Si S 


k = rf 


如 上 述 代码 所 示 ， 标 粗 的 即 为 更 改 的 内 容 ， 其 中 [groups] 定 义 了 一 个 组 并 命名 为 admin; 
中 表示 目录 权限 分 配 ， 意 味 看 将 整个 项 目的 根 目录 权限 分 配给 admin 组 ， 并 指定 该 组 用 户 能 
够 可 斌 可 写 ;，[/bbs] 与 [/user] 分 别 表示 配置 bbs 和 user 目录 权限 ， 这 里 将 使 用 单个 用 户 进行 配 
置 ， 并 给 出 相应 的 权限 。 

将 文件 保存 后 ， 还 需要 在 主 配 置 文件 中 开 司 权限 配置 。 打 开 svnserve.conf 文件 ， 找 到 # 
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authz-db = authz 市 点 ， 将 # 注 释 符 与 容 格 删 除 ， 这 样 束 完成 了 权限 配置 。 由 于 还 需要 禁止 匿 
名 用 户 浏 览 版 本 库 ， 所 以 还 需要 将 svnserve.conf 文件 中 的 # anon-access = read 配置 项 更 改 为 
anon-access = none (HAARET). PIE, SVN 权限 配置 束 完 成 了 。 

需要 说 明 的 是 ，SVN 只 是 实现 版 本 控制 ， 但 对 于 团队 开发 来 说 ， 版 本 控制 只 是 文件 管 
理 的 一 部 分 。 要 真正 实现 团队 开发 及 测试 ， 还 需要 配置 团队 调试 环境 。 有 具体 细节 可 参阅 本 书 
附录 B。 


Q 提示 : TrotoiseSVN 在 保存 用 户 及 密码 后 ， 没 有 提供 清除 用 户 与 密码 的 功能 ， 但 可 以 通过 删除 SVN 程序 
数据 的 方式 清空 用 户 数 据 与 密码 ， 该 文件 位 于 (以 Windows 7 为 例 ) Ci:\Users\Administraton\ 
AppData\Roaming\Subversion\ 目录 ， 将 该 目录 下 的 authz 文件 删除 即 可 。 


1.5 小结 


本 章 一 开始 话 细 回顾 了 PHP 的 发 展 历史 ， 了 解 PHP 技术 展 趋 势 。 接 看 讲解 了 PHP 环境 
的 搭建 ， 本 章 介 绍 的 环境 搭建 注重 的 是 实践 性 ， 采 用 了 高 效 的 一 键 安装 包 的 方式 进行 讲解 ， 
确保 每 个 读者 能 够 正确 安装 ， 迅 速 进入 学 习 状 态 。 

本 章 还 详细 介绍 了 PHP 主流 IDE， 方 便 读者 选择 开发 工具 。 最 后 深入 介绍 了 版 本 控制 
系统 的 使 用 ， 因 为 MVC 的 开发 是 分 工 明 确 的 ， 深 入 学 习 SVN 对 于 一 名 合格 的 PHP 程序 员 
来 说 非常 有 必要 ， 由 于 篇 幅 所 限 本 章 并 没 讲解 Linux 下 的 SVN 搭建 ， 读 者 可 以 参阅 SVN È 
方 网 站 帮助 文档 。 无 论 是 Windows 还 是 Linux 下 的 SVN， 命令 使 用 和 权限 配置 都 是 一 样 。 

本 章 所 讲解 的 内 容 比较 理论 化 ， 下 一 章 将 深入 学 习 PHP 的 面 问 对 象 。 面 问 对 象 开 发 是 
MVC 设计 的 基础 ， 读 者 需要 深入 掌握 。 
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内 容 提 要 

面向 对 象 编程 不 是 一 门 技术 ， 而 是 一 种 编程 思想 ， 它 与 菜 种 特定 的 技术 不 存在 联系 ， 一 
门 高 级 的 现代 化 语言 最 重要 的 一 个 因素 就 是 看 它 是 不 是 支持 面向 对 象 ， 比 如 CH, Java, C++ 
等 都 能 够 很 好 地 支持 面向 对 象 编程 ; 尽管 每 种 技术 在 实现 面向 对 象 时 不 尽 相 同 ， 但 都 有 封 
装 、 继 承 、 多 态 等 特性 。PHP 从 5.0 版 本 开始 已 经 正式 提供 了 面向 对 象 编程 技术 ，PHP 的 面 
向 对 象 编程 技术 借鉴 了 Java 众多 思想 ， 能 够 满足 大 型 软件 的 开发 需要 ， 这 是 PH 一 个 重要 
的 进步 。 本 章 将 会 首先 介绍 面向 对 象 技 术 ， 帮 助 读者 树立 面向 对 象 编 程 思想 ; 接着 详细 介绍 
PHP 对 面向 对 象 编 程 的 支持 . 

学 习 目 标 

o 理解 面向 对 象 编程 思想 。 
理解 Class 关键 字 的 意义 。 
理解 类 方法 与 函数 的 区 别 。 
理解 类 的 继承 作用 。 
掌握 实例 类 与 静态 类 的 使 用 。 
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2.1 面向 对 象 介绍 


面 问 对 象 编程 思想 最 早出 现在 20 世纪 60 年 代 。C++、Objective-C 等 高 级 语言 的 出 现 ， 
丰富 了 面向 对 象 编程 思想 。 以 Java 为 代表 的 全 面向 对 象 编程 技术 ， 近 年 来 已 广泛 地 应 用 在 
大 型 工业 化 的 软件 设计 中 ， 面 向 对 象 编程 搁 术 是 计算 机 编程 史上 的 一 个 里 程 碑 ， 它 让 程序 更 
加 容易 维护 、 重 用 、 部 署 等 。 面 问 对 象 类 似 于 例子 操作 ， 程 序 员 不 必 知 道 类 功能 具体 实现 过 
程 ， 只 需要 知道 类 提供 了 哪些 可 用 的 方法 (操作 实体 )， 然 后 进行 调用 即 可 ， 这 就 意味 着 该 
类 是 可 以 多 处 调用 、 多 处 使 用 的 。 这 里 所 说 的 多 处 调用 与 传统 函数 在 方式 上 是 类 似 的 ， 只 是 
面向 对 象 提供 的 多 重用 性 更 加 高 效 和 彻底 。 如 果 具 备 面 向 过 程 设计 基础 (C、PHP 4.X、 
Asp, VB 等 )， 理 解 PHP 面 问 对 象 编程 并 不 是 一 件 困 难 的 事 。 下 面 首 先 对 类 的 几 个 最 重要 概 
念 进行 一 个 人 简单 的 介绍 。 更 详细 的 使 用 将 在 后 面 的 内 容 中 结合 示例 代码 进行 讲解 。 

1. 类 

类 是 面 回 对 象 编程 中 最 重要 的 概念 ， 所 谓 的 类 和 我 们 日 常 中 见 到 的 事物 类 是 一 样 的 。 比 
如 地 球 上 有 动物 类 、 植 物 类 等 物种 ， 动 物 类 中 又 包含 了 人 、 猫 、 狗 等 ， 这 些 就 是 类 的 实体 对 
象 ， 这 里 就 可 以 看 出 所 谓 的 类 只 不 过 是 人 们 用 于 方便 识别 物种 的 一 个 方法 ， 它 并 不 具体 存 
在 ; 但 是 动物 类 中 的 人 、 猫 、 狗 等 是 真实 存在 的 ， 可 以 完成 特定 的 事情 ; 同样 ， 在 软件 设计 
中 类 也 是 用 于 总 结 、 归 类 代码 的 一 个 称呼 ， 它 并 不 能 实现 具体 的 功能 ， 但 类 包含 的 方法 却 可 
以 完成 具体 的 功能 。 

2. 对象 

设计 好 的 类 在 被 计算 机 编译 后 ， 便 被 分 配 内 存 区 、 地 址 、 资 源 等 。 系 统 需要 使 用 类 ， 就 
需要 找到 相应 的 内 存 地 址 ， 然 后 进行 编译 ， 但 还 没有 触发 相应 的 功能 ， 处 于 等 待 接受 任务 状 
态 ， 被 分 配 到 内 存 中 的 资源 句柄 束 称 为 对 象 ， 这 里 可 以 简单 地 理解 为 类 是 一 个 真实 存在 的 文 
件 ， 它 存在 于 人 硬 检 上 ， 而 对 象 是 一 个 存放 于 内 存 中 的 抽象 资源 ， 可 以 随时 销毁 。 通 常情 况 下 
类 与 对 象 是 一 一 对 应 的 ， 一 个 没有 实例 化 〈 没 有 生成 对 象 ) 的 类 是 不 能 使 用 它 所 包含 的 功能 
模块 (静态 类 除外 )。 

2. DZ 

前 面 所 讲 的 功能 模块 是 类 包含 的 具体 功能 代码 ， 也 就 是 类 方法 。 虽 然 类 方法 的 声明 在 每 
种 编程 语言 上 会 有 不 一 样 的 声明 方式 ， 但 通常 情况 下 类 的 方法 就 是 第 见 的 功能 函数 ， 将 一 个 
功能 函数 放 到 类 中 ， 就 会 成 为 该 类 的 方法 ， 一 个 类 需要 实现 哪些 功能 ， 通 常 都 需要 在 类 方法 
中 实现 ， 这 就 是 类 的 封闭 性 。 它 可 以 让 开发 者 像 使 用 功能 函数 一 样 方便 赋 参 、 调 用 等 ， 还 可 
以 作为 类 的 接口 对 其 他 方法 进行 约束 、 规 范 等 ， 从 这 里 就 可 以 看 出 ， 类 的 主要 作用 就 是 归 
类 、 合 并 传统 编程 中 的 函数 、 变 量 等 。 

4. 成 员 变量 

类 成 员 属 性 也 称 成 员 变 量 ， 它 通常 用 于 接收 外 部 数据 或 将 内 部 数据 返回 ;上 所谓 的 成 员 属 
性 和 常见 的 变量 是 一 样 的 ， 它 们 都 有 自己 的 数据 类 型 、 赋 值 方式 等 。 类 成 员 变 量 与 传统 变量 
在 作用 上 都 是 一 样 的 ， 只 是 类 成 员 变 量 只 能 作用 于 类 中 ， 并 且 受 类 的 权限 修饰 符 限 制 。 类 成 
员 变 量 在 类 被 实例 化 后 就 称 为 类 成 员 属性 ， 事 实 上 均 是 指 同 一 概念 。 
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5. 继承 

继承 是 面 问 对象 编程 中 最 重要 的 一 种 思想 ， 面 问 对 象 编程 思想 核心 融 是 继承 。 所 谓 的 继 
承 顾名思义 束 是 “接受 旧 的 产生 新 的 ”这 束 意 味 看 莉 类 具备 旧 类 的 全 部 功能 ， 继 承 丰 祝 了 
面向 对 象 编程 思想 ， 使 得 程序 设计 更 加 便捷 、 局 效 。 本 书 重 点 介绍 的 MVC kuk 
于 继承 实现 的 ， 在 此 读者 只 需要 徊 单 理 解 即 可 。 


2.2 PHP 面向 对 象 基础 


PHP 面 问 对 象 编程 锥 形 在 PHP 4.X 时 已 经 形成 ， 但 直到 PHP 5.0， 才 真正 能 够 完美 地 文 
持 面 问 对 象 编程 。 PHP 5.0 面向 对 象 不 像 过 去 那样 仅仅 是 一 个 继承 的 功能 ， 而 是 提供 了 封 
装 、 拆 箱 、 类 保护 、 接 口 、 抽 象 层 等 类 似 于 Java 的 高 级 功能 。PHP 的 面向 对 象 设计 出 色 地 
继承 了 传统 PHP 的 简洁 、 高 效 ， 能 够 为 Web 开发 提供 一 流 的 性 能 与 效率 。 接 下 来 将 首先 从 
PHP 类 设计 开始 学 习 ， 如 果 读 者 已 经 具备 面 癌 对 象 编程 思想 可 以 略 过 本 章 ， 但 如 果 是 初学 者 
或 者 未 接触 过 PHP 编程 的 读者 ， 务 必 阅 读本 章 内 容 。 


2.2.1 class 关键 字 


class 直译 成 中 文 即 是 “类 ”“ 类 别 ” 的 症 思 。 前 面 已 经 讲述 过 ， 类 是 面 同 对 象 设 计 的 基 
础 单元 。 在 PHP 中 ， 使 用 关键 字 “class” 作 为 类 体 的 声明 ，PHP 编译 器 一 旦 过 到 该 关键 
字 ， 怠 会 按 处 理 面 癌 对 象 的 方式 为 程序 分 配 相关 的 资源 。 如 以 下 代码 所 示 。 


<?php 


ki 


class technology { 
function PHP (){ 


EE EE E oNN e 
} 
function Jsp(){ 


recur “Jaor 
} 


function asp(){ 


cerir Vago” e 
| 
} 
smyClass=new technology () ; 


echo "我 想 学 " . SmyClass->PHP () ; 
?> 


上 述 代码 运行 结果 为 “我 想 学 php”。 代 码 中 的 “class” 关 键 字 为 类 声明 ， 然 后 是 类 的 名 

称 ， 类 名 称 后 面 的 大 括号 “{}” 即 为 类 的 实体 。 所 有 的 功能 代码 必须 写 于 类 实体 内 。 在 进行 
类 声明 时 ， 需 要 注意 以 下 内 容 。 

> 类 名 称 必须 符合 规范 ， 通 常情 况 下 使 用 一 个 具有 意义 的 英文 单词 作为 类 名 ， 如 果 类 

作为 一 个 单独 文件 存在 ， 通 常 类 名 与 文件 名 同名 ， 建 议 使 用 “类 名 .class.php” 的 文 
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件 命 名 方式 。“class 123ab”, “class .abc _ class” 这些 类 命名 方式 都 是 非法 的 ， 不 能 使 
HI. "7 abc class”, “ab_c class” 这 些 类 命名 方式 是 合法 的 ， 但 不 建议 使 用 。 

> 类 体 中 的 “function ”关键 字 即 为 类 方法 声明 ， 普 通 的 函数 存在 于 类 体 中 ， 束 称 为 类 
方法 。PHP 为 弱 类 型 语言 ， 对 类 的 数据 类 型 不 需要 手动 声明 。PHP 对 变量 是 严格 区 
分 大 小 写 的 ， 但 对 类 名 、 方 法 、 函 数 名 称 是 不 区 分 大 小 写 的 。 

> 尽量 避免 在 类 方法 中 输出 HIML， 这 会 在 一 定 程 度 上 破坏 类 的 通用 性 。 

前 面 使 用 了 一 个 稍 单 的 示例 ， 介 绍 了 一 个 简单 的 次 构 成 ， 一 个 类 能 够 完成 什么 功能 ， 通 

用 性 如 何 ， 需 要 通过 类 的 成 员 来 实现 ， 下 和 面 将 进一步 介绍 类 的 成 员 组 成 。 


2.2.2 类 中 的 成 员 


一 个 功能 完整 的 PHP 类 体 都 会 包含 许多 类 成 员 ， 类 成 员 的 设计 决定 了 类 的 意义 与 功 
能 ， 除 了 前 面 提 到 的 类 方法 ， 常 见 的 PHP 类 成 员 还 包括 构造 函数 、 成 员 变 量 、 类 常量 、 释 
构 函 数 等 ， 下 面 将 分 别 介绍 。 

1. 构造 函数 

构造 函数 是 一 个 比较 特殊 的 函数 ， 它 用 于 类 对 象 初始 化 时 的 默认 传 参 。 一 个 比较 复杂 的 
功能 类 ， 需 要 实现 的 功能 非常 多 ， 程 序 员 在 调用 类 时 往往 需要 处 理 一 大 堆 默 认 参 数 ， 使 用 构 
造 函 数 束 可 以 在 实例 化 类 时 直接 以 news (参数 1 参数 2) 的 方式 进行 赋值 。 如 以 下 代码 所 示 。 


<?php 


class classilf{ 
private $name; 
private $action; 
/大 大 
* 构造 函数 
大 
* Qparam unknown type $_name 
er 
function _ construct ($_name, $_action) 
{ 
$sthis->name=$_ name; 
$this->action=$_action; 
} 
// 目 定义 方法 
function action(){ 
echo $this->name.Ş$this->action; 
} 
} 


Claes class2! 

public $name; 

publie $accion, 

// 目 定义 方法 

function action(){ 

echo $this->name.$this->action; 
} 

} 
// 有 构造 函数 的 类 
ER 
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ŠŚclassl->action()}; 


// 没 有 构造 函数 的 类 
$sclass2=new class2(); 
sclass2->name=" 李 开 消 "; 
$class2->action=" 在 写 代码 "， 
$class2->action(); 


?> 


上 述 代 码 共 有 两 个 类 ， 分 别 为 classl 和 class2， 它 们 输出 的 结果 都 为 “ 李 开 涌 在 写 代 


。classl 类 使 用 了 __construct 声明 了 一 个 构造 函数 ， 构 造 函 数 共 有 两 个 参数 $_name、 


$_action， 并 指定 两 个 参数 的 值 赋 给 成 员 变 量 $name 和 $action， 以 便 其 他 类 方法 进行 调用 ， 
这 两 个 参数 在 该 类 中 的 作用 束 是 强制 实例 化 时 必须 赋 参 ， 盏 则 编译 器 将 会 报错 。 通 过 构造 水 
数 能 够 为 类 的 完整 性 提供 了 保证 。 


class2 没有 使 用 构造 测 数 ， 虽 然 也 实现 了 和 classl 相同 的 功能 ， 但 在 实例 化 类 时 需要 进 


行 额外 的 属性 属 赋值 ， 代 码 也 多 了 3 行 。 上 述 示例 只 是 简单 地 演示 构造 函数 的 功能 ， 在 类 功 
能 复杂 的 情况 下 ， 构 造 函 数 可 以 方便 开 及 人 员 初 始 化 类 。 此 外 ，PHP 构造 函数 除了 使 用 


construct 进行 声明 外 ， 还 可 以 使 用 与 类 同名 的 方法 作为 该 类 构造 冰 数 。 


2. 成 员 变 量 


成 员 变 量 也 称 为 成 员 属 性 ， 它 是 由 普通 的 变量 加 上 访问 修饰 符 而 构成 的 ， 成 员 变 量 和 普 


通 变 量 一 样 只 用 于 临时 存放 数据 ， 在 PHP 类 设计 中 类 变量 还 用 于 接收 外 部 数据 和 反馈 方法 
运行 结果 。 如 以 下 代 公 所 未 。 


<?php 

class myClasst 
public SNowTime,; 
Ee 


public function getTime (){ 
// 使 用 成 员 变 量 返 回 数据 


$this->NowTime=date ("Y-m-d h:i:s"); 


| 
public function MyLike()t 
// 输 出 类 成 员 变 量 数 据 
echo "我 喜欢 " .Sthis->1L1ike， 


} 


/ /实例 化 类 

sclass=new myClass () ; 
$class->getTime () ; 
echo $class->NowTime; 


melass -like "PHPT, 
$class->MyLike(); 
?> 


3. 释 构 函数 
PHP 类 构造 函数 一 旦 被 初始 化 ， 编 详 旧 即 会 在 操作 系统 内 存 区 中 建立 资源 堆栈 ， 和 直到 程 


序 发 出 释放 资源 命令 ， 才 会 释放 该 类 所 创建 的 资源 。PHP 使 用 _destruct0 函 数 进行 释 构 ， 它 
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的 功能 是 跟 __ construct 构造 疝 数 取 反 值 的 ， 如 以 下 代 公 所 示 。 


<?php 
class class1{ 
private $name; 
private $action; 
/** 
* 构造 函数 
大 
* Qparam unknown type $_name 
E 
function _ construct ($_name, $_action) 
{ 
Ş$this->name=$_name; 
$this->action=$_action; 
} 
// 目 定义 方法 
function action(){ 
echo $this->name.$this->action; 
} 
// 释 放 类 变量 name HI action 的 资源 
EMCE LON —_ Cesrirucr (i 
$sSthis->name; 
şthīis->action; 


} 


} 

// 有 构造 函数 的 类 

aes lw eles | 
SE eene DE 

?> 


如 上 述 代 人 码 所 示 ， 使 用 __destruct0 函 数 清除 由 构造 函数 创建 的 $name、$action 资源 ， 这 
过 程 是 目 动 化 的 ，PHP 不 返回 任何 消息 。 事 实 上 PHP 的 构造 函数 在 实际 应 用 开发 中 是 不 
需要 的 ， 因 为 PHP 内 置 了 垃圾 回收 机 制 (garbage collection )， 就 算 不 手动 进行 释放 ，PHP 也 
能 自动 清理 残留 的 内 存 资源 ， 保 证 程序 的 运行 效率 。 


gi 


2.2.3 ”实例 化 类 


一 个 PHP 类 被 new 后 就 称 为 实例 化 类 。 实 例 化 后 的 类 称 为 对 象 ， 程 序 员 面 对 医 对 象 不 
需要 奏 看 基体 的 实现 过 程 ， 只 需要 知道 类 的 公开 接口 《成 员 属性 、 公 开 的 方法 等 ) 加 可 以 完 
成 类 的 使 用 。 这 也 融 意 味 看 ， 一 个 封闭 的 类 对 开 友 人 员 来 六 是 封 箱 的 〈 像 封 钱 在 箱子 里 的 物 
体 )， 普 通 的 程序 员 没 必要 修改 封装 好 的 类 ， 这 束 很 好 地 保证 了 PHP 类 的 通用 性 和 重用 性 ， 
一 个 设计 优秀 的 类 甚至 可 以 应 用 在 任何 PHP 应 用 中 。 下 而 分 别 使 用 两 个 文件 演示 PHP 实例 
化 类 。 

<?php 

//MyClass.php 文件 

class MyClass{ 

/** 
* 输出 天 气 信 息 


大 


* Qparam unknown type S$str 


* Qreturn unknown 
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SE 
function Weather ($str){ 
return "现在 天 气 为 : " .$str; 
} 
} 


接着 创建 index.php 文件 ， 实 例 化 前 面 创建 的 MyClass， 如 以 下 代码 所 示 。 
<?php 


//index.php 

require ("MyClass.php"); 
sclass=new MyClass () ; 

echo S$class->Weather ("Z3"); 
?> 


在 index.php 中 只 需要 包含 MyClass.php 文件 ， 即 可 以 实现 面 癌 对 象 编程 。 开 发 人 员 不 
必 知 道 类 功能 实现 过 程 ， 只 需要 调用 接口 ， 即 可 完成 相应 功能 的 调用 。 如 图 2-1 所 示 。 


<2php 
Eel Eel My Less, php”: 
fcClaassnew MyClass{):; 
echo Scleaa-3 
Œ MyClass: :Weather (unknown type satt) unknown 


te aik d 


Location; PHPDocurnenti 
Class Mat Jass 


function Veathertunknomamn type $str) 
输出 和 天气 信息 
Returns: 

unknown 


图 2-1 调用 类 公开 接口 


2.3 ”类 中 的 方法 


PHP 关 特 性 ， 还 能 够 实现 类 的 高 度 封 装 。 关 方法 中 比较 重要 的 几 个 概念 分 别 为 方法 的 参数 、 
访问 修饰 符 、$this 关键 字 等 ， 下 面 分 别 介绍 。 


2.3.1 方法 的 参数 


PHP 之 所 以 灵活 强大 ， 一 部 分 原因 来 日 于 其 内 部 海量 的 也 数 库 。PHP 的 函数 库 能 够 让 程 
序 员 以 较 少 的 代码 实现 高 效 、 强 大 的 功能 。 一 名 优秀 的 PHP 程序 员 ， 理 解 并 熟悉 PHP 内 置 
的 函数 库 是 必 不 可 少 的 。PHP 内 置 函 数 中 ， 多 数 都 是 需要 传 参数 值 的 〈 即 赋 参 )。 只 有 正确 
地 赋 参 ， 编 译 器 才能 得 到 函数 运算 的 结果 。 同 样 ， 在 目 定 义 类 方法 中 也 可 以 定义 方法 参数 ， 
以 便 开 发 人 员 传 进 正确 的 参数 ， 实 现 方法 的 功能 运算 。PHP 方法 参数 定义 与 函数 参数 定义 是 
相同 的 ， 如 以 下 代码 所 示 。 
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<?php 
class MyClasst 
private stimes; 
romet LON CONSTE N) 
SEniS >ELnes=are( NV nme Hiev)? 
} 


/** 
* 目 定 方法 
大 
x Qparam unknown type smyname 
x @param unknown type $action 
E 
function action myname array (), action, Saddr=ss T HYH 
foreach ($myname as $value) { 
Ş$name.=Ş$value; 


} 


return HI. Srhis-ztimes, ln. Sname. än count (ëmenamei I A". Saction. WG, 
".Saddress; 
} 
} 


sclass=new MyClass () ; 
echo $class->action (array ("#7 ., ", "小 明 "), "下 在 汐 冰 。")， 


如 上 述 代 码 所 示 ，action 方法 一 共有 3 个 参数 : 其 中 第 1 个 参数 声明 为 一 个 数组 ， 在 
PHP 中 不 需要 强制 声明 变量 类 型 ， 编 详 占 会 目 动 世 配 ， 所 以 这 里 使 用 了 一 个 array0 作 为 默认 
值 ， 以 便 调 用 的 程序 员 知 道 这 里 需要 传递 一 个 数组 参数 ， 第 2 个 参数 为 字符 如 变量， 并且 没 
AWE: B3 个 参数 是 一 个 已 经 赋 有 默认 值 的 参数 ， 作 用 与 普通 函数 中 的 堆 认 参数 值 是 一 
样 的 ， 即 在 调用 时 该 参数 赋值 是 可 选 的 。 在 实例 化 类 时 ， 和 直接 调用 action 方法 并 赋值 即 可 。 
最 终 运 行 结束 如 下 。 

[2011-12-26 09:23:21] 李 开 涌 ， 小 明 总 共 2 个 人 正在 溜冰 。 地 点 为 : 广场 


2.3.2 ”方法 的 返回 信 


成 员 方 法 使 用 return 返回 方法 的 运算 结果 ， 只 有 使 用 return 返回 ， 调 用 的 程序 才能 接收 
到 结果 值 ， 使 用 echo 输出 运算 结果 ， 调 用 的 值 将 会 为 空 。 前 面 已 经 讲述 过 PHP 声明 变量 是 
不 需 指定 类 型 的 ，PHP 类 方法 返回 值 同样 不 需要 指定 数据 类 型 ， 这 点 对 于 从 其 他 语言 转 到 
PHP 的 开发 者 而 言 尤 其 需要 注意 ， 但 是 开发 人 员 也 可 以 使 用 类 型 声明 将 结果 强制 转换 为 相应 
的 数据 类 型 ， 如 以 下 代码 所 示 。 


<?php 
class MyClass{ 


function actionī1(){ 
return array 人 
ere 
We 
"星期 三 "=>"C#"， 
ERR E 
"星期 五 "=>"cn 
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} 


ERRI, actioni 方法 将 返回 一 个 数组 ， 外 部 程序 在 获取 结果 时 ， 可 以 通过 数组 下 标 
获取 a 到 相应 的 值 。 在 复杂 的 应 用 中 ， 通 第 需要 多 个 方法 参与 运算 ， 最 后 才 返 回 运 算 结 末 ， 如 
以 下 代码 所 示 。 


<?php 


class MyClass{ 
function actionī1(){ 
ore CE be HE 
PECUen SEnis ee Deene 7 
} 
private function action2($str){ 
ET 
} 


} 

Şclass=new MyClass () ; 

echo Sclass=>actiomi (); 

DS 

上 述 代码 使 用 了 2 个 方法 共同 运算 ， 最 后 由 actionl 方法 返回 运算 结果 。 在 功能 复杂 的 
关中， 这样 设 计 是 经 稼 用 到 的 。 


2.3.3 ”访问 修饰 从 


访问 修饰 符 可 以 控制 类 成 员 的 权限 域 ， 它 是 面 癌 对 象 设计 中 最 经 党 使 用 到 的 技术 ， 在 前 
HREF, BAZEI public, private 关键 了 学 ， 这 些 关 键 学 束 是 类 成 员 〔( 包 含 成 员 变 量 
和 成 员 方 法 ) 访问 修饰 待 。 使 用 访问 修饰 符 可 以 完善 类 的 设计 ， 增 强 类 的 封装 性 、 严 讶 性 和 
安全 性 。 一 个 良好 的 类 体 设 计 ， 应 该 具备 完善 的 访问 修饰 ， 以 防止 被 恶意 调用 ， 破 坏 程序 的 
通用 性 。 在 PHP 中 共有 3 个 访问 修饰 符 ， 分 别 如 下 。 

> public (ETHAN): 该 修饰 从 是 PH 类 设计 中 的 默认 修饰 从 ，public 其 有 完全 开放 

的 权限 ， 这 意味 看 类 成 员 可 以 被 类 其 他 成 员 或 类 对 得 使 用 。 

> protected (FARI): 表示 该 成 员 只 能 在 本 类 或 子 类 中 进行 调用 。 

> private (MA): 表示 只 能 在 本 类 中 调用 ， 不 能 在 子 类 中 调用 。 

如 以 下 代码 所 示 ， 只 有 actionl 方法 公开 可 调用 ， 调 用 其 他 两 个 方法 或 属性 时 ，PHP A 
译 器 将 会 报错 ， 但 可 以 在 内 部 进行 调用 。action3 方法 的 修饰 符 为 protected， 所 以 可 以 在 子 类 
(sonMyClass) 中 进行 调用 。 

<?php 


class MyClass{ 
private $name; 
function action1(){ 
ee 
cecurcn $chis=>act Lon? (98T) 7 
} 
private function action2 (S$str)t 


return $str.": 下 午 我 在 写 文章 " ; 
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} 


protected function action3 () { 


return "有 晚上 我 在 看 电视 "; 
} 


} 


class sonMyClass extends MyClass { 
function myoff (){ 
return hie accions H, Aa 
} 
} 


Şclass=new sonMyClass ();}; 
echo $class->myoff(); 


?> 


运行 结果 为 “晚上 我 在 看 电视 ， 一 会 不休 筷 ”， 读 者 可 以 坚 试 在 子 类 中 调用 其 方法 ， 观 
察 运行 结果 有 哪些 变化 。 


2.3.4 $this 关键 字 
$this 关键 学 是 PHP 类 设计 中 比较 特殊 的 变量 ， 它 只 能 在 类 内 部 使 用 ， 用 于 在 方法 中 调 
用 成 员 变 量 或 其 他 成 员 方 法 。 前 面 的 内 容 中 已 经 多 次 出 现 过 $this KEF, HEA OAA A 
生 ， 如 以 下 代码 所 示 ， 演 示 了 $this 关键 字 的 典型 应 用 。 
<?php 
class MyClasst 
private Ş$name=1; 
/大 大 
ns TED DI Dr 
AA 
public function actionī1(){ 
return $this->action2(); 


b 


private function action2(){ 
if ($this->name==1) { 
return "REN 1"; 
}else { 
return CHAE 
} 
} 
} 
sclass=new MyClass () ; 
echo Selass=> actioni () p 


?> 


上 述 代 码 运 行 结果 为 “ 络 末 值 为 1”。 值 得 需要 注意 的 是 ， 引 用 闫 变量 时 $this 关键 学 后 
不 能 接 $ 符 号 。 例 如 $this->$name 是 错误 的 ， 而 $this->name 是 正确 的 。 


2.4 类 的 继承 
继承 可 以 让 类 设计 变 得 更 加 合理 ， 面 问 对 象 设 计 中 最 重要 的 一 个 概念 束 是 继承 ， 使 用 继 


50 国 国 


第 2 章 面向 对 象 基础 


承 可 以 让 类 的 作用 得 到 无 限 延 伸 。 所 谓 的 继承 顾名思义 束 是 子 类 继承 父 类 ， 开 发 人 员 可 以 开 
发 出 一 个 新 的 闪 ， 然 后 继承 到 已 经 存在 的 茯 。 旧 的 奖 吏 是 父 类 《或 称 基 关 )， 新 的 基 叫 做 子 
关 (或 称 派生 类 )。 下 面皮 和 完 介 绍 最 第 见 的 子 类 继承 父 尖 ， 了 解 类 设计 的 层 钦 天 系 。 


2.4.1 子 类 继承 父 类 

从 现实 的 角度 考虑 ， 假 设 宕 要 为 “动物 ”创建 一 个 类 ， 暂 时 称 其 为 Animals 类 ; biz 
程序 的 功能 越 来 越 多 ， 现 在 需要 创建 一 个 类 表示 狗 ， 和 暂时 称 其 为 Dog: 下 面 还 可 能 有 猫 
类 、 猪 类 每。 这 些 类 都 有 一 个 共同 点 ， 它 们 都 是 需要 食物 (Food)、 有 睡觉 《Sleep)、 行 走 
(Walk) 等 。 

这 些 共同 点 即 为 动物 的 共性 ， 如 果 不 使 用 继承 功能 ， 需 要 在 每 个 类 中 创建 同样 的 食物 、 
睡觉 、 行 走 ， 而 使 用 了 继承 ， 由 于 在 子 类 中 可 以 完全 使 用 父 类 中 所 有 公开 的 属性 和 方法 。 这 
就 意味 着 子 类 同样 有 了 父 类 的 同 点 ， 轻 松 地 实现 了 面向 对 象 的 封装 性 和 重用 性 ， 如 以 下 代码 
HIZR o 


<?php 


class Animals{ 

protected function Food(){ 
recura Sizmi eat 

} 

BrOoOEeeced uncion Slees()! 
er EE 

} 

protected function Walk()t{ 


Ie a 2 AE 
} 
} 
GE 
<?php 
几 / 十 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 
//| 狗 ， 继 承 于 Animals 类 
人 二 二 = 一 一 二 一 一 一 一 一 一 一 一 一 二 


class Dog extends Animals 
public function YellowDog (){ 
echo "黄色 的 狗 3 点 钟 " . Sthis->Walk () ."6 点 名 钟 " .Sthis->Sleep() ."8 点 钟 " .Sthis->Food() ; 
} 
public function BlackDog () { 
echo "黑色 的 狗 4 点 钟 " .Sthis->Walk () ."7 点 名 钟 " .Sthis->Sleep () ."9 点 钟 ".$this->Food (); 
} 
public function White (){ 
echo "黄色 的 狗 5 点 钟 " . Sthis->Walk () ."8 KKE .$this->Sleep ()."10 点 钟 ".$this->Food (); 
} 
} 
$class=new Dog () ; 
$class->YellowDog(); 
?> 


如 上 述 代码 所 示 ， 共 有 两 个 类 :; Animas 为 其 类 ， 定 义 了 3 个 方法 用 于 表示 动物 所 共有 
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的 特性 ; Dog 类 为 子 类 ， 使 用 extends 关键 字 指 明了 继承 于 Animals X, Dog 类 包含 3 个 方 
法 ， 用 于 表示 狗 的 颜色 ， 由 于 狗 都 有 共性 ， 所 以 直接 调用 父 关 中 已 经 存在 的 方法 即 可 。 运 行 
结果 如 下 。 

黄色 的 狗 3 点 钟 活动 中 . . .6 点 钟 睡觉 中 .. .8 点 钟 吃饭 ... 

PHP 的 类 继承 关系 是 无 限制 的 ， 以 上 述 代码 为 例 ，Dog 同样 可 以 再 被 其 他 类 继承 ， 一 旦 
被 继承 相应 的 子 类 将 会 包含 Dog 所 公开 的 方法 和 属性 。 需 要 说 明 的 是 类 继承 关系 必须 要 注意 
访问 修饰 符 的 作用 ， 和 否则 将 会 造成 权限 不 足 ， 继 承 将 失败 。 


Q 提示 : extends 关键 字 用 于 类 的 继承 ， 一 旦 被 继承 ， 实 际 上 就 相当 于 实例 化 一 次 ， 所 有 在 继承 关系 中 的 
类 被 extends 后 就 相当 于 被 new 一 次 。 


2.4.2 ” 重 与 父 类 中 的 成 员 属性 

通常 会 在 父 类 中 为 一 些 重 要 的 成 员 属 性 设置 默认 值 ， 在 子 类 继承 时 将 会 得 到 父 类 中 公开 
的 成 员 属 性 值 ， 这 如 意味 看 子 关 可 以 像 实 例 化 对 象 一 样 获取 或 重 写 父 关 中 的 成 员 属 性 ， 以 便 
实现 程序 运行 过 程 中 的 特殊 需要 ， 如 以 下 代码 所 示 。 


<?php 

EE 

// | 天 气 基 类 

//+--------- 

class WeatherBase { 
// 国 家 
protected $country=" 中 国 "; 
// 城 市 
protected Soait we | 
// 温 度 
protected Stemperature=16; 
EH 


protected Sweather; 


class sonWeather extends WeatherBase { 
public function Forecast () { 

Sthis->country=" 北 京 "; // 修 改 城市 名 
$this -temperature 3; // EME 
$this->weather=" %53"; // 为 天 气 赋值 
/ /返回 数组 
return array 人 

WEE le = 

Wolo El SO ey, 


"temperature"=>Ş$this->temperature, 
"weather"=>$this->weather, 
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// 调 用 

$sclass=new sonWeather () ; 
$rows=$class—->Forecast (); 
// 输 出 数组 数据 

var_dump ($rows); 

?> 


如 上 述 代码 所 示 ， 在 父 类 中 定义 了 4 个 成 员 属 性 ， 并 且 都 是 公开 可 继承 的 。 在 子 类 的 
Forecast 方法 中 ， 通 过 改写 父 类 中 的 成 员 属 性 ， 以 便 实现 天 气 了 预报 的 变更 ， 最 后 使 用 数组 组 
织 数 据 并 返回 ， 运 结果 如 以 下 代码 所 示 ， 


Ea da Te er Se (| 
mm en Strinoiëi N A 


2.4.3 final KRF 


前 面 已 经 介绍 过 父 类 是 可 以 无 限制 被 子 类 继承 的 ， 父 类 方法 同样 也 是 能 够 被 子 类 继承 或 
调用 的 。 出 于 类 的 严 刘 性 或 功能 需要 ， 某 些 类 或 方法 不 希望 被 继承 调用 ， 这 时 就 需要 使 用 
PHP 内 置 的 final 关键 字 了 。final 可 以 很 好 地 控制 类 继承 范围 ， 防 止 被 无 限 层次 调用 ， 如 以 
下 代码 所 示 。 


<?php 
final class BaseClass{ 


protected function actionl () { 
Karona o e 
} 
} 
class myClass extends BaseClass { 
function result (){ 
return "调用 的 结果 为 : ".$this->action1(); 
} 
| 
?> 
<?php 
// 调 用 
Şclass=new myClass () ; 
echo Selases-z—resultih: 
EE 


上 述 代码 使 用 关键 字 final 声明 了 基 类 BaseClass 为 终极 类 ， 不 允许 被 其 他 类 继承 ， 如 果 
强行 继承 ， 编 译 占 将 会 报错 ， 如 以 下 代 人 码 所 示 。 
Fatal error: Class myClass may not inherit from final class (BaseClass) in 


PHPDocument1 on line 12 
final 关键 字 不 仅 可 以 声明 类 为 终极 类 HAR DE), Ate nf Elend TAN 
终极 方法 ， 如 以 下 代码 所 示 。 


<?php 


class BaseClass{ 
// final 声明 为 终极 方法 
final function actionl(){ 
EE 
} 
| 
class myClass extends BaseClass { 


// 试 图 重 写 父 类 中 的 actioni () 方法 
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function actionī1(){ 
EE 

} 

} 

?> 

<?php 

// 调 用 

Şclass=new myClass () ; 

echo $class->actionl (); 


?> 

如 果 强 行 调用 ， 编 译 器 将 会 报错 ， 如 以 下 代码 所 示 。 

Fatal error: Cannot override final method BaseClass::actionl() in PHPDocumentl1 on 
line 12 


使 用 final 关键 字 可 以 更 严谨 地 控制 美的 作用 域 ， 使 闫 设计 更 加 安全 与 合理 。 


25 静态 类 成 员 


PHP 静态 类 成 员 主要 包括 静态 方法 、 静 态 属 性 以 及 类 钊 量 。 其 中 静态 成 员 方 法 和 静态 属 
性 使 用 static 关键 字 进 行 定 义 。 所 谓 的 静态 类 成 员 它 是 类 设计 中 比较 特殊 的 结构 ， 普 通 的 类 
方法 是 类 对 象 〈 类 实例 ) 的 一 部 分 ， 但 静态 类 成 员 它 只 属于 类 体 的 一 部 分 ， 在 类 实例 化 对 象 
中 资源 是 空 的 ， 编 译 器 不 会 为 静态 类 成 员 创 建 内 存 扒 栈 ， 所 以 在 调用 静态 类 成 员 时 ， 不 需要 
实例 化 ， 只 需要 知道 静态 类 成 员 所 在 的 类 名 即 可 。 

例如 代码 Myclass::name。Myclass 是 一 个 显 式 的 类 名 ， 没 有 经 过 new 实例 化 ， 直 接 调 
用 头 中 的 静态 属性 即 可 。 在 类 或 继承 的 类 中 调用 静态 类 成 员 时 ， 不 能 再 使 用 $this KEF. 
但 PHP 提供 了 另外 两 个 重要 的 关键 字 ， 用 于 在 静态 方法 内 调用 其 他 静态 类 成 员 ， 下 面 分 别 


人 “人 


2.5.1 static 关键 字 


在 成 员 变 量 前 加 上 statice 关键 字 即 成 为 了 静态 类 变量 ， 它 的 定义 方式 与 普通 的 区 成 员 变 
量 并 没有 较 大 区 别 ; 如 果 在 类 成 员 方法 前 加 上 statice 关键 字 ， 那 么 该 方法 即 成 为 了 静态 方 
法 。 如 有 条 一 个 类 中 既 有 实例 化 方法 ， 也 有 静态 方法 ， 那 么 在 静态 方法 中 将 不 能 直接 使 用 $this 
天 键 字 进行 调用 。 下 面 通 过 代码 活 示 statice 关键 字 的 使 用 。 
<?php 
class BaseClass{ 
static $name=" J"; 
static $time; 
// 更 改 静态 变量 Stime 的 值 
function _ construct  (){ 
self: :Ş$time=date ("Y-m-d H:i:s"); 


} 

/ 大 大 
* 静态 方法 
大 
E 


static function action(){ 
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return "[".self::$time."]".self::S$name." 下 在 写 代 码 "， 


} 

E 

// 调 用 

echo BaseClass::action();} 

如 上 述 代码 所 示 ， 使 用 关键 字 statice 声明 了 两 个 成 员 变 量 和 1 TH Dr, AARE 
这 3 个 类 成 员 都 是 静态 化 的 。 前 面 已 经 讲述 过 ， 一 旦 类 的 成 员 被 静态 化 ， 它 束 不 属于 类 对 象 
的 一 部 分 ， 所 以 在 调用 时 直接 使 用 显 式 的 “类 名 :: 毅 态 类 成 员 ” 方 式 调用 即 可 。 但 是 ， 由 于 
构造 函数 并 非 鹏 态 成 员 ， 圾 要 在 对 象 中 进行 实例 化 ， 它 才 会 补 调 用 ， 所 以 上 述 代 人 码 的 构造 函 
数 将 会 处 于 失效 状态 ， 运 行 结果 如 以 下 代码 所 示 。 


[] 李 开 涟 正在 写 代 码 
事实 上 ， 在 调用 静态 类 成 员 时 ，PHP 提供 了 非常 弹性 的 方式 。 开 发 人 员 可 以 在 非 静态 方 
法 中 使 用 实例 化 的 方式 调用 毅 态 类 成 员 ， 如 以 下 代码 所 示 。 
<?php 
class BaseClass{ 

St ale mame mA 

static $time; 

// 更 改 静 态 变量 Stime 的 值 

DE ER te SE ebe () 

EE EE 


/大 大 
* 静态 方法 
SECH 
Sacie funeocnon accron() 1 
return "[".self::$time."]".self::S$name." 下 在 写 代 码 "， 
} 
} 
P 
<?php 
// 调 用 
$sclass=new BaseClass () ; 
echo $class->action(); 
D 
上 述 代 人 码 使 用 了 类 对 象 ， 它 不 仅 能 够 实例 化 类 ， 还 能 调用 包括 静态 类 成 员 在 内 的 类 成 
员 。 事 实 上 在 调用 实例 类 成 员 时 ， 开 发 人 员 同 样 可 以 使 用 “class::action0” 调 用 实例 类 成 
员 ， 但 需要 配置 php.ini 的 错误 级 别 ， 否 则 将 会 产生 “Non-static” 错 误 。 经 过 改造 ， 上 述 代 
但 中 的 构造 函数 将 会 被 执行 ， 静 态 闫 成员 也 会 被 调用 。 
值得 说 明 的 是 ， 虽 然 PHP 提供 了 多 种 调用 静态 类 成 员 的 方式 ， 但 在 实际 应 用 开发 中 ， 通 
香 使 用 “类 名 :: 静 态 类 成 员 ” 的 方式 调用 静态 类 成 员 ; 而 使 用 new 关键 学 调用 实例 类 成 员 。 


2.5.2 ”访问 静态 类 成 员 (self::parent::) 
在 前 而 的 例子 中 使 用 sef 关键 学 访 问 娘 态 类 成 员 ， 而 不 能 使 用 $this 关键 字 。 在 继承 的 类 
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体 中 同样 使 用 sellf 访问 子 关 中 的 静态 闫 成 员 ， 而 访问 父 类 的 静态 类 成 员 时 需要 使 用 parent X 


键 字 。 如 以 下 代码 所 示 。 


class BaseClass{ 
protected static $name; 
protected static $age=26; 
BEOESEGEESEGC scacie SpororessLlon, 
/ /静态 方法 
protected static function action (){ 
self: :Sŝprofession=" HMM EFE"; 


perur €u date (Jy=-m- d HeT: S4). ne 


} 

// 非 静态 方法 

protected function action2 () { 
Seer a e 


class sonClass extends BaseClass { 
private static saddress Y IITA GR"; 


private static sachool FE AFT 


public static function result (){ 
// 静 态 方 法 中 调用 非 静 态 方法 
SBaseClass=new BaseClass(); 
ŞBaseClass->action2(); 
// 返 回 数组 


return array 人 


/ / EH parent KEFY HRH WEA E 


"名 称 "=>Parent : : $name, 

uw parent: 0 

WIER EE action), 

// 访 问 本 类 的 静态 成 员 ， 使 用 self 关键 字 
"住址 "=>self::S$address, 

"毕业 学 校 "=>self::$school, 


p> 

<?php 

// 调 用 
$rows=sonClass::result ();，; 
var_dump ($rows); 

2> 


: :Şprofession; 


如 上 述 代码 所 示 ， 基 关中 定义 了 3 个 静态 类 变量 、1 个 静态 类 方法 以 及 1 个 实例 类 方 
法 ;在 子 类 中 调用 普通 的 方法 时 ， 需 要 将 基 类 实例 化 (事实 上 这 样 的 设计 是 不 合理 的 ， 在 实 
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际 使 用 时 可 以 在 实例 化 过 程 中 重 写 即 可 ， 这 里 主要 用 于 演示 ， 方 便 读者 更 进一步 认识 静态 方 
法 与 非 静态 方法 的 区 别 )。 在 调用 基 关 静态 类 成 员 时 使 用 parent， 但 调用 本 类 的 静态 类 成 员 时 
依然 使 用 self。 


2.5.3 ”类 常量 


让 类 设计 中 ， 有 些 数 据 是 恒久 不 变 的 ， 通 第 这 些 数 据 只 用 于 呈现 固定 的 数据 。 比 如 声明 一 
个 圆周 率 值 3.14， 为 了 数据 的 完整 性 ， 束 没有 必要 使 用 类 变量 进行 声明 。 因 为 那样 惑 意味 看 程 
序 可 以 随时 更 改 数 据 ， 这 明显 不 符合 设计 要 求 ， 使 用 类 第 量 整 可 以 很 好 地 解决 上 述 问 题 。 类 当 
量 和 普通 常量 一 样 ， 它 只 用 于 保存 数据 ， 一 旦 定义 束 不 能 在 程序 运行 过 程 中 修改 第 量 的 值 。 在 
定义 类 第 量 时 ， 只 能 使 用 const 关键 子 进 行 定义 ， 不 能 使 用 define0 定 义 ， 如 以 下 代码 所 示 。 


<?php 


class BaseClasst{ 
const MAM HZH: 
const AGE=26;}; 
const ADDRESS="] RA IMT"; 
static function action () { 


return array 人 
"姓名 "=>self: :NAME, 
"年 龄 "=>self: :AGE, 
"地 址 "=>self: :ADDRESS,， 


class sonClass extends BaseClass { 

function result () { 
return parent::action(); 

} 

| 

p> 

<?php 

// 调 用 

$sclass=new sonClass ()，} 

var_ dump (sclass->result ()); 

2 


如 上 述 代码 所 示 ， 在 基 类 中 使 用 const 关键 子 定 义 了 3 个 类 第 量 、1 个 静态 方法 ， 在 子 
类 中 实现 了 一 个 普通 方法 。 细 心 的 读者 已 经 看 出 ， 在 内 部 调用 类 第 量 时 和 调用 前 态 变 量 一 样 
使 用 self 和 parent 关键 子 ， 事 实 上 类 的 常量 束 是 类 有 的 一 个 只 读 的 琵 态 类 成 员 ， 在 类 外 部 调用 
时 间 样 可 以 套用 “类 名 :: 沼 量 ” 格 式 进行 调用 但 不 能 修改 )。 


D 提示 : 在 定义 类 常量 时 必须 要 赋值 ， 类 常量 不 需 设置 访问 修饰 符 ， 它 是 全 局 的 ， 不 受 访问 修饰 符 限制 。 
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2.6 小结 


本 章 介 绍 了 PHP 面 癌 对 象 基础 ， 这 些 基础 知识 是 使 用 PHP MVC 的 基础 ， 如 果 和 掌握 不 
牢 ， 后 面 的 MVC 内 容 将 会 学 得 非常 壮 兰 ， 这 点 尤其 需要 刚 接 触 PHP 的 读者 注意 。PHP 面 问 
对 象 是 由 传统 的 PHP 面 癌 过 程 升级 而 来 的 ， 如 果 读 者 没有 接触 过 PHP 的 面向 过 程 设计 ， 这 
一 章 阅 读 起 来 可 能 比较 困难 。 如 果 有 必要 ， 该 者 可 以 查阅 相关 资料 ， 了 解 PHP 面 癌 过 程 开 
发 。 如 PHP 结构 、 变 量 、 数 组 、 函 数 等 。 下 一 章 将 继续 深入 PHP 面 癌 对 象 ， 帮 助 读 者 完全 
理解 PHP 面 癌 对 象 设计 。 
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内 容 提要 

在 PHP5 中 完善 了 类 结构 设计 ， 增 加 了 许多 实用 功能 ， 相 信 通 过 前 面 的 学 习 ， 读 者 已 经 
能 够 理解 面向 对 象 设计 的 一 些 重要 特性 。 一 个 功能 强大 的 类 不 仅 能 够 做 到 条 理 分 明 、 结 构 清 
晰 ， 最 重要 的 是 要 实现 代码 高 度 可 重用 。 这 也 是 所 有 面向 对 象 设计 语言 所 共有 的 特性 

在 大 型 软件 设计 中 ， 设 计 模 式 是 一 个 非常 重要 的 概念 ， 只 有 合理 的 软件 架构 ， 前 期 完善 
的 UML 流程 图 ， 在 软件 开发 的 各 个 阶段 ， 才 能 游刃有余 。 提 及 设计 模式 ， 就 很 有 必要 提 及 
软件 设计 中 的 接口 、 抽 和 象 、 多 态 、 n 等 技术 了 ， 这 在 C++ 编程 中 应 用 得 非常 广泛 ，PHP 从 
5.0 开始 也 提供 了 这 些 编程 概念 。 本 草 将 会 以 上 一 章 的 内 容 为 基础 ， 继 续 讲解 PHP 面向 对 象 
开发 中 的 高 级 特性 ， 0 E 
的 知识 ， 布 望 读 者 能 够 牢固 等 握 。 

学习 目标 

@ 进 一 入 巩固 面 辣 对 象 开发 拉 术 。 

© 理解 PHP 类 克隆 。 
理解 PHP 接口 。 
理解 抽象 类 。 
掌握 PHP 异常 处 理 。 
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3.1 类 对 象 的 克隆 (clone ) 


前 而 已 经 提 到 过 ， 一 个 类 体 被 实例 化 后 就 是 一 个 对 象 ， 类 对 和 象 可 以 直接 使 用 ， 如 果 和 需要 
多 次 调用 类 体 ， 就 需要 不 断 地 重复 实例 化 (new)， 这 对 于 严谨 的 设计 来 说 是 不 友好 的 ， 程 序 
的 可 维护 性 会 变 兰 。PHP 5 提供 clone 关键 字 用 于 元 隆 类 对 和 象 ， 元 隆 后 的 结果 将 会 得 到 类 对 
象 指针 AD 值 )， 而 不 是 一 个 燃 体 的 副本 ， 从 而 提高 运行 靳 。 如 以 下 代码 所 示 。 

<?ph 

e 七 es 七 工 

{ 

和 
} 


class test2 

{ 
public $testlobj; 
public oce en SE 


} 


Stesti I = new test), Z/testi DIr 
Stesti new Cestliiz test T Aana 
Stest2_l->restloobo] = Seasels 

stest2 2 = clone S$Stest2 1;// 复 制 test2_1 对 象 


$test2_1l->testlobj->test1V = ' 吃 早餐 '， 
SEE 

EE EE EE 
GE 


在 PHP5 Dn AWER ZR, nl Dm done) DM rä fr, BERNIA 
的 克隆 在 实际 开发 中 应 用 得 比较 少 ， 所 以 这 里 不 做 介绍 。 


3.2 ”类 接口 ( interface ) 


前 面 介 绍 的 内 容 侧重 于 类 体 的 功能 实现 ， 读 者 学 握 前 面 的 内 容 ， 基 本 上 能 够 满足 利 见 的 
PHP 开发 。 因 为 这 些 内 容 部 是 基于 功能 运算 的 ， 所 以 在 一 些小 的 项 目 中 将 会 接触 得 非 第 频 
繁 。 接 下 来 将 讲解 PH 在 大 型 应 用 开发 中 经 种 采用 的 闫 接口 技术 ， 关 接口 技术 是 软件 设计 
模式 的 范畴 ， 它 不 涉及 功能 的 运算 ， 但 却 约 束 看 功能 运算 结 采 ， 类 接口 拉 术 在 复杂 的 应 用 中 
应 用 得 非常 广泛 ， 几 乎 所 有 面 辣 对 象 语言 都 提供 了 类 接口 编程 技术 ， 关 接口 技术 不 仅 能 够 请 
足 大 型 软件 开发 灵活 分 工 的 需求 ， 还 能 让 软件 架构 变 得 有 条 有 理 。 下 面 首 先 理解 类 接口 中 几 


个 重要 概念 。 


3.2.1 接口 的 意义 


在 一 些 复 杂 、 大 型 的 软件 设计 中 ， 需 要 由 众多 的 功能 类 构成 ， 这 些 功能 关 需 要 由 多 个 程 
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序 员 共 同 完成 ， 其 中 软件 架构 师 需要 制定 设计 模 式 、 团 队 分 工 等 工作 。 这 会 剖 来 编程 统一 性 
的 问题 ， 如 何 保证 关键 的 功能 能 够 很 好 地 实现 ， 这 就 需要 一 个 能 够 统筹 兼顾 的 设计 接口 ， 在 
PHP 中 能 够 采用 的 方案 有 许多 种 ， 其 中 类 接口 是 最 常用 的 一 种 。PHP 类 接口 能 够 限制 需要 
实现 的 功能 列表 ， 方 便 地 约束 项 目 程序 员 按 既定 的 设计 模式 进行 ， PHP 类 接口 实现 过 程 如 
图 3-1 所 示 。 


Imain 接口 


linterface Imainf 
public function Ton) 
public function fnat) 
public function Tote di 


i 


T 
[|elass Head implements ji ' [elass Foot implements lmaint 
Irmaint publice furection faii 
publice function fna) echo “HHT: 
echo “ti ër t 
} publice function fnat 
publice function fnz0! echo "Ef": 
echo Yin”: t 
] publice function fnisd}f 
publice function fr sg echo" Vir ed 
echo "ik Sur d: i] 
|] 
Head 类 实现 Imain 接口 Foot 类 实现 Imain 接口 


图 3-1 PHP 定义 与 实现 接口 


如 图 中 所 示 ， 接 口 是 独 立 的 ， 一 个 合法 的 接口 ， 可 以 在 多 个 类 体 中 实现 。 接 口 不 负 贡 功 
能 的 实现 ， 它 就 像 一 个 界面 ， 定 义 了 需要 实现 的 类 成 员 ， 这 些 成 员 在 实现 接口 的 功能 类 中 必 
须要 实现 ， 人 否则 程序 将 会 停止 运行 并 报错 。 这 就 保障 了 程序 的 完整 性 ， 让 程序 员 知道 目 己 十 
要 实现 的 功能 和 业务 。 


3.2.2 ”定义 接口 


接口 不 是 类 体 ， 不 能 在 接口 类 体 中 定义 已 经 实现 好 的 方法 ， 在 定义 接口 时 使 用 interface 
关键 字 进 行 定义 ， 接 口 名 称 通常 按照 “I+ 接口 名 称 ”的 格式 进行 命名 。 接 口 类 体内 所 定义 的 
成 员 必 须 为 类 体 方法 ， 而 不 能 是 类 成 员 属 性 ， 如 以 下 代码 所 示 。 
<?php 
/** 
zs SC I Ee 
wy 
interface Imyinterface{ 
function addi (); 
function add2 ($s); 
function add3 ($t=0); 


| 


定义 接口 成 员 时 需要 注意 几 点 : 接口 成 员 必须 具有 全 局 访问 权限 ， 不 能 添加 访问 修饰 
TT, 接口 成 员 不 能 使 用 钟 量 、 静 态 方法 等 关 必 性， 接口 成 员 不 能 定义 为 构造 图 数 。 

接口 可 以 像 类 一 样 可 以 继承 ， 在 继承 关系 的 接口 中 ， 子 接口 将 会 得 到 父 接口 的 全 部 成 
员 ， 如 以 下 代码 所 未 。 
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<?php 

/大 大 

zs SC MIT Tmintertace 

大 

E 

interface Imyinterface{ 
function addi () ; 
function add2 ($s); 
function add3 ($t=0); 

| 

// 接 口 的 继承 

interface Imys extends Imyinterface { 
function del: 
function edit () ; 
function update ($str); 
function select ($id, $str); 


Es 


3.2.3 ”实现 接口 〈implements ) 


前 面 已 经 介绍 过 接口 只 能 进行 功能 定义 ， 但 并 不 能 实现 具体 的 功能 。 要 实现 接口 中 定义 
的 方法 ， 必 须 在 普通 的 功能 类 中 进行 实现 。 在 实现 接口 时 ， 只 需 使 用 “implements+ 接 口 名 
称 ” 的 方式 创建 一 个 普通 芒 能 类 即 可 。 如 以 下 代码 所 示 。 


<?php 

/** 

| ee 

大 

a 

interface Imyinterface{ 
function Ee ie GE 
function add2 ($s); 
function add3 ($t=0); 


i 

DS 

<?php 

// 实 现 接口 


class MyClass implements Imys { 
function addi (){ 


} 
function add2 ($str){ 


} 
function add3 (St=0) { 


} 


?> 


从 MyClass 类 中 可 以 看 出 ， 实 现 接口 和 类 继承 非常 相似 ， 只 是 关键 字 不 一 样 而 已 。 类 继 
承 使 用 extends 关键 字 ， 而 实现 接口 使 用 implements 关键 了 学 。 同 时， 需要 在 实现 类 中 实现 接 
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口中 所 有 已 定义 的 成 员 方 法 (不 能 遗漏 任意 一 个 )。 一 旦 实现 了 接口 中 所 定义 的 成 员 方法 
(可 以 是 空 方法 )， 其 他 的 成 员 属性 可 以 是 普通 的 类 成 员 。 如 果实 现 继承 关系 中 的 接口 ， 需 要 
同时 实现 子 接口 与 父 接 口中 所 有 定义 的 成 员 ， 如 以 下 代码 所 示 。 
<?php 
/** 
* 定义 接口 Imyinterface 
SCH 
interface Imyinterface{ 
function addi (); 
function add2 ($s); 
function add3($t=0); 


| 
// 接 口 的 继承 
interface Imys extends Imyinterface { 
runer ion Cel) z 
function edit (); 
function update ($str); 
EE select (Sic, HSEE)S 


?> 
<?php 
class MyClass implements Imys { 


function addi () { 


} 
function add2 ($str){ 


} 
function add3 (St=0) { 


function del () { 


} 


function edit  (){ 


} 
function update(Ssir F Ae 
return $str; 


} 
function select (ŝid=0, ŝstr=" ZFR") { 
return Sid SStr; 


smys=new MyClass ();}; 
echo Smys->select () ; 


2 
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使 用 implements 关键 宇 不 仅 可 以 实现 一 个 接口 ， 还 可 以 在 一 个 美 中 同时 实现 多 个 接口 ， 
这 在 接口 众多 的 程序 架构 中 是 非常 有 用 的 。 使 用 implements 实现 多 个 接口 非常 灵活 和 简单 ， 
只 需要 在 各 接口 乙 间 使 用 “,” 隔 开 即 可 ， 如 以 下 代 但 所 示 。 
<?php 
// 接 口 2 
interface Imyinterface{ 
function addi (); 


function add2 ($s); 
function add3 ($t=0); 


} 
/7 接 回 2 
interface Imys { 
E Eet EE eh Ce Ee e 
NIKE Gu Ee 
function update ($str); 
function select ($id, $str); 


R 

<?php 

// 实 现 多 个 接口 

class MyClass implements Imyinterface, Imys{ 
function add1 (){ 


} 
function add2 ($s){ 


} 
function add3 ($t=0){ 


} 
function dell 


} 


function edit (){ 


} 
function update ($str){ 


} 
function select ($id, $str){ 
return Ş$str.$id; 
} 
} 
smy=new MyClass (); 
echo Smy->select (3," 星 期 ") ; 


> 


3.3 ”抽象 类 与 抽象 万 法 


PHP 抽象 类 与 抽象 方法 是 主流 MVC 框架 所 采用 的 设计 模式 ， 抽 象 类 能 够 实现 接口 类 似 
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的 功能 ， 但 抽象 类 与 接口 是 两 个 不 同 的 设计 模式 。 一 个 可 控 、 可 定制 的 开发 项 目 ， 它 的 设计 
模式 应 该 是 清晰 和 合理 的 ， 抽 象 类 与 接口 相 结合 能 够 让 PHP 设计 模式 更 加 合理 及 规范 。 


3.3.1 理解 抽象 概念 


在 上 一 市 中 介绍 了 PHP 接口 在 设计 模式 中 的 实际 应 用 ， 接 口 在 大 型 软件 设计 中 具有 非 
名 重要 的 作用 ， 虽 然 接口 能 够 规范 和 统一 程序 的 各 个 层次 设计 ， 但 由 于 接口 上 只 能 定义 程序 抽 
象 界 面 ， 而 不 能 提供 具体 的 功能 ， 所 以 接口 的 灵活 性 加 大 打折 扣 。 抽 和 象 类 与 接口 非常 类 似 ， 
它们 者 能够 提供 界面 设计 模式 ， 但 抽象 类 可 以 在 抽象 类 体 中 实现 普通 的 类 成 员 ， 这 也 是 抽象 
类 比 接口 更 灵活 的 地 方 。 

抽象 类 是 为 类 继承 而 设计 的 ， 所 定义 的 抽象 方 法 必须 在 子 类 中 实现 ， 在 定义 抽象 方法 时 
需要 加 上 abstract 关键 字 ， 然 后 写 上 普通 的 成 员 方 法 《不 需要 有 具体 的 功能 代码 )。 从 这 里 融 可 
以 看 出 抽 和 象 闪 和 接口 的 共同 点 束 是 一 个 界面 可 以 极 多 个 子 关 实现 ， 这 对 于 大 型 的 软件 设计 模 
式 是 非常 有 用 的 ， 后 面 章 节 将 要 学 习 的 PHP MVC 框架 大 多 都 是 采用 PHP 抽象 类 构建 的 。 

在 日 党 生活 中 “抽象 ”是 一 种 事物 总 体 描 述 ， 不 会 具体 到 事物 的 本 质 。 在 PHP 设计 中 
也 是 同样 的 过 理 ， 一 个 抽象 类 只 需要 让 开发 人 员 知 道 该 类 的 意义 即 可 ， 人 至 于 需要 实现 什么 样 
的 功能 ， 那 就 要 看 抽象 实现 关 的 需求 了 。 

抽象 类 不 能 个 实例 化 ， 但 所 定义 的 成 员 属 性 变量 ) 可 以 在 子 类 的 类 对 和 象 中 进行 调用 。 
抽象 类 与 接口 都 是 设计 模式 中 重要 的 技术 ， 可 以 简单 地 理解 为 抽象 类 等 于 普通 功能 类 加 上 类 


接口 〈abstract=class+interface ) 。 


3.3.2 ”定义 抽象 类 和 方法 (abstract) 


抽象 类 和 抽象 方法 的 关键 子 为 abstract， 除 此 之 外 抽象 尖 和 普通 功能 类 并 没有 多 大 的 区 
别 。 在 PHP 中 ， 抽 象 闫 不 能 定义 为 空 类 或 普通 类 ， 抽 象 类 全 少 需 要 提供 一 个 抽象 方法 ， 这 
样 抽象 闫 才能 被 实现 。 以 下 代码 是 一 个 简单 的 抽象 闫 结构 ， 共 备 了 人 稍 单 意义 上 的 CURD Co 
建 数 据 、 更 新 数据 、 读 取 数 据 、 删 际 数据 〉 操 作 概 仿 ， 代 人 码 如 下 。 


<?php 


abstract class Baseclasst 
// EW, MATIK 
abstract function query (); 
// 插 入 ， 抽 和 象 方法 
abstract function insert (); 
// EP 205 Dr 
abstract function update (); 
/ /删除 ， 抽 象 方法 
abstract function deleteii: 
/ /数据 库 连 接 ， 普 通 类 方法 
public $mysqli; 


public function conn (){ 
Şthis->mysqli=new mysqli("localhost", "root", "root", "testdatabases"); 
if (S$Sthis->mysqli->connect error) die(" H 4# J: ".S$this->mysqli->connect 
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EE 
EE 
return $this->mysqli; 
} 
// 关 闭 数据 库 连 接 ， 普 通关 方法 
public function Closeconn () { 
return $this->mysqli->close(); 
} 
} 
?> 


上 述 代 码 中 Baseclass 类 是 一 个 抽象 类 ， 并 定义 了 4 个 抽象 方法 和 两 个 普通 方法 ， 最 终 抽 
象 方法 会 被 要 求实 现 ， 而 普通 的 方法 也 能 直接 使 用 ， 可 见 抽象 类 相对 于 接口 显得 更 加 灵活 。 使 
用 抽象 类 能 够 方便 地 实现 常见 的 “抽象 类 工厂 ”设计 模式 ， 该 模式 是 MVC 框架 的 基础 。 


3.3.3 ”使 用 抽象 类 


抽象 闫 一 般 用 于 设计 模 陈 的 最 确 层 。 尽 管 抽 象 基 内 可 以 包含 普通 的 闫 成员， 但 由 于 抽 和 象 
类 是 不 能 实例 化 的 ， 所 以 在 使 用 抽象 类 时 ， 首 先 必须 派生 一 个 子 类 ， 然 后 在 子 类 中 实现 抽象 
类 中 所 定义 的 抽象 方法 。 以 下 代码 将 会 实现 BaseClass 抽象 类 中 所 定义 的 抽象 方法 ， 并 完成 
数据 库 CURD 操作 。 


<?php 


class MyClass extends BaseClass{ 
function _ construct () { 
Ş$this->conn(); 
} 
public $lasError; 
public $sql; 
// 实 现 奉 询 
function query (){ 
Şres=$this->mysqli->query (Ş$this->sql); 
if (S$Srows=$res—->fetch array (MYSQLI_ASSOC)){ 
return S$rows; 
}else { 
Şthis->lasError=$this->sql; 
return false; 
} 
Şthis->CloseConn(); 
} 
// 实 现 插 入 


function insert (){ 


} 
// 实 现 更 新 
function update () { 
if (Şsmt=Şthis->mysqli->prepare (Ş$this->sql)){ 
Şthis->CloseConn ();} 


Eeturpn rue: 
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}else { 
$this->lasError=$smt->error; 
return false; 

k 

| 
// 实 现 删除 
function qelete() { 
if (Şsmt=Şthis->mysqli->prepare (Ş$this->sql)){ 
Sthis->CloseConn ();}; 
recurm Crue; 

}else { 

Ş$this->lasError=$smt->error; 


return false; 


} 
PS 
<?php 
// 使 用 
Şobj=new MyClass () ; 
SE El SELECT = EE HEEN 
$rows=$o0bj->query () ; 
if (SrowsS) { 

var_dump ($rows); 
}else { 

EE EE 


} 


实现 抽象 类 和 普通 类 的 区 别 之 处 在 于 必须 要 全 部 实现 抽象 方法 ， 其 他 的 类 成 员 与 普通 的 
类 成 员 并 没有 区 别 。 抽 象 类 是 可 以 被 多 个 子 类 进行 继承 的 ， 在 实际 应 用 开发 中 通常 使 用 抽象 
类 和 方法 定义 程序 的 通用 模块 ， 派 生子 类 可 以 根据 业务 需求 重 写 抽象 类 中 的 抽象 方法 ， 实 现 
类 的 高 度 重 用 。 


3.3.4 ”接口 与 抽象 类 的 区 别 


通过 前 面 的 学 习 ， 相 信 读 者 已 经 对 接口 和 抽象 类 有 了 更 具体 认识 。 接 口 和 抽象 类 都 是 面 
问 对 象 编程 中 一 种 重要 的 设计 模式 ， 在 一 些 主流 的 面 癌 对 象 编程 技术 中 应 用 得 非常 广泛 ， 虽 
然 接 口 与 抽象 类 在 使 用 上 类 似 ， 实 现 的 目的 也 相同 ， 但 它们 之 间 还 存在 一 些 明 显 区 别 ， 下 面 
将 介绍 接口 与 抽象 类 之 同 的 区 别 。 

1. 接口 与 抽象 类 的 共同 点 

接口 是 引用 类 型 ， 是 一 套 行 为 和 规范 ， 它 用 于 摘 述 软件 上 层 设计 的 一 个 界面 ， 告 诉 软件 
设计 者 “我 能 做 什么 ? ”。 抽 和 象 类 是 类 的 功能 抽象 化 ， 需 要 通过 继承 关系 才能 人 确定 抽象 类 实 
现 的 功能 ， 它 们 之 间 都 是 软件 设计 的 主流 模式 ， 同 时 也 存在 一 些 共同 点 ， 分 别 如 下 。 

> 接口 和 抽象 类 都 不 能 被 实现 化 ， 接 口 需要 使 用 implements 实现 ;而 抽象 类 使 用 普通 

类 的 extends 关键 字 继 承 。 
> 接口 和 抽象 类 都 包含 看 未 实现 的 方法 声明 。 
> 派生 类 必须 实现 未 实现 的 方法 ， 抽 和 象 类 是 抽象 方法 ， 接 口 则 是 所 有 成 员 。 
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2. 接口 与 抽象 类 的 区 别 
抽象 关 是 对 对 象 的 抽象 ， 可 以 把 抽象 类 理解 为 类 的 摘 述 。 而 接口 只 是 一 个 行为 的 规范 或 
规定 。 两 者 之 间 的 区 列 概 括 如 下 。 
> 抽象 类 不 能 被 密封 ,但 接口 可 以 。 
> 抽象 类 实现 的 具体 方法 默认 为 虚 的 ， 但 实现 接口 的 类 方法 上 默认 为 实 的 。 
> 抽象 类 必须 为 在 该 类 的 基 类 列表 中 列 出 所 有 成 员 以 便 让 实现 类 实现 ， 但 接口 允许 空 
方法 。 


3.4 类 的 异常 


无 论 是 早期 的 PHP 编译 右 ， 还 是 现在 的 ZEND 处理 引 区 都 内 置 实用 的 脚本 异常 信息 处 
理 机 制 。 程 序 员 可 以 通过 改变 php.ini 配置 文件 改变 异常 信息 级 别 (error_reporting)， 实 现 查 
看 及 控制 PHP 脚本 运行 过 程 中 的 寞 和 党 信息 。 最 狐 版 的 Zend Engine 2.0 还 提供 了 
triger_error、 set_error_handler、error_log 疯 数 ， 并 统一 封装 到 了 Exception 类 ， 通 过 这 些 郴 数 
和 类 ， 开 发 人 员 束 可 以 方便 地 在 PHP 程序 中 处 理 异 党 信息 了 。 


3.4.1 Exception 类 


PHP 5 提供 了 Exception 寞 常 处 理 类 ， 开 发 人 员 可 以 使 用 try 语句 块 捕捉 程序 运行 过 程 中 的 寞 


HARE (BUG), Exception 异常 处 理 类 封装 了 捕捉 异常 信息 所 需要 的 类 成 员 ， 代 码 如 下 所 示 。 
<?php 


class Exception 


{ 


protected $message = 'Unknown exception'; // IAA 

Btokeekerd Zecgde se D: // Bn DV STEI 
protected $file; // RETE RFA 
protected $line; // REE ARET S 
function _ construct ($message = null, $code = 0); 

final function getMessage(); // 返回 异常 信息 

final function getCode () ; // 返回 异常 代码 

Al tl rt EE // 返回 发 生 异 常 的 文件 名 
final function getLine(); // 返回 发 生 异 常 的 代码 行 号 
final function getTrace();}; Ii backtrace O) AA 
final function dt Te a Neo no) Vp Te /| 


/* 可 重 载 的 方法 */ 

EE On — Ee bere dek // 可 输出 的 字符 串 
} 
GE 


Exception 类 是 一 个 异常 信息 处 理 的 基 类 ， 开 发 人 员 可 以 方便 地 继承 和 重 写 类 成 员 ， 实 
现 自 定义 异常 处 理 。 如 以 下 代码 所 示 。 


<?php 
/** 
* 自 定义 一 个 异常 处 理 类 
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E 
class MyException extends Exception 


{ 
// 重 定义 构造 器 使 message 变 为 必须 被 指定 的 属性 
public function _ construct ($message, $code = 0) { 


// 目 定义 的 代码 
// 确保 所 有 变量 都 被 正确 赋值 


parent::_ construct (Smessage，S$Scodqe) ; 


// 目 定 义 字 符 串 输出 的 样式 
EE EE EE 
return CLASS . ": [{$this->code}]: {$this->message}\n"; 


public function customFunction() { 
echo "A Custom function for this type of exception\n"; 


SE 


3.4.2 ”使 用 try、catch、throw 语句 


Exception 黄 封 疼 了 处理 异 音信 息 的 类 成 员 ， 开 友人 员 可 以 使 用 try 语句 块 调用 Exception 
类 捕捉 到 的 异 背 信息， 如 以 下 代码 所 示 。 


<?php 


$name = "Name"; 


//check if the name contains only letters, and does not contain the word name 


Li (oreg mecca! /| a2/1i", Şmame)) 
{ 


throw new Exception ("$name contains character other than a-z A-Z"); 


} 


if (strpos (strtolower ($name), 'name') !== FALSE) 


{ 


throw new Exception ("$name contains the word name"); 


} 


echo "The Name is valid"; 


} 


catch (Exception $e) 


{ 


throw new Exception ("insert name again",0, $e); 


} 


catch (Exception $e) 


{ 
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if ($e->getPrevious()) 
d 


echo "The Previous Exception is: ".$e->getPrevious ()->getMessage ()."<br/>"; 


} 
echo Vihe mxesorion lss EE 


} 


用 try 语句 块 捕捉 异常 信息 非常 简单 和 直观 ， 在 其 他 一 些 成 熟 的 面向 对 象 设计 语言 (如 
C#、Java、C++ 等 ) 已 经 应 用 得 非常 广泛 ，try 语句 是 PHP 5.0.1 新 增 的 语法 ， 同 时 还 提供 了 
throw 关键 字 用 于 中 断 进 程 并 抛 出 异 利 信息 ， 如 以 下 代码 所 示 。 


<?php 
EE 
require_once ('cmd_php5/Command.php'); 


class CommandManager { 
EIERE EE ongs p 


function getCommandObject ($cmd) { 
spath = "{$this->cmdDir}/{$cmd}.php"; 
if (!file_exists(Ş$path)) { 
throw new Exception ("Cannot find $path"); 
} 
require_once $path; 
iE (lelass exists (Sanch) | 
throw new Exception("class S$cmd does not exist"); 


} 


$class = new ReflectionClass ($cmd); 
if (!$class->isSubclassoOf (new ReflectionClass ('Command'))) { 
throw new Exception("$cmd is not a Command"); 


} 


return new $cmd(); 


} 


PHP 异 第 处 理 机 制 在 实践 应 用 开发 中 并 不 是 必需 的 ， 但 一 个 民 好 的 程序 员 应 该 具备 排 合 
错误 的 能 力 ， 使 用 PHP 提供 的 卉 党 处 理 功能 ， 能 够 让 开发 人 员 及 早 地 发 现 程序 BUG， 方 便 
后 续 维 护 。 


意 : 过 多 地 使 用 try 语句 块 会 导致 PHP 程序 执行 效率 降低 ， 如 果 没 有 必要 ， 在 实践 的 生产 环境 中 应 该 
尽量 减少 使 用 try 语句 块 ， 通 常情 况 下 建议 只 在 文件 处 理 、 远 程 网 络 处 理 等 功能 时 才 使 用 try 语句 块 。 


3.5 ”小结 


本 草 接 看 上 一 划 的 内 容 ， 继 续 深 入 PH 和 面 同 对 象 技 术 ， 结 合 多 个 示例 全 面 讲解 了 PHP 
面向 对 象 的 高 级 用 法 ， 这 些 技术 都 是 MVC 设计 的 重要 内 容 ， 特 别 是 PHP 的 接口 、 抽 象 类 等 
内 容 ， 读 者 务必 掌握 牢固 ， 本 章 内 容 部 分 示例 引用 于 PHP 官方 手册 ， 读 者 如 需要 更 多 帮 
助 ， 可 以 前 往 www.php.net 进 行 搜索 个 阅 。 
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内 容 提要 

MVC 是 软件 设计 工程 中 的 灵魂 ， 它 能 够 让 开发 人 员 养 成 恨 好 的 编程 习惯 ， 简 化 开发 流 
程 ， 提 高 效率 和 项 目 质 量 。MVC 编程 并 不 是 必需 的 ， 但 如 果 需 要 进入 PHP 编程 更 高 境界 ， 
接触 MVC 设计 模式 是 必需 的 。 当 然 ， 既 然 选择 阅读 本 书 ， 笔 者 当然 知道 你 的 选择 。 从 本 章 
开始 ， 读 者 将 会 学 习 几 和 套 完 善 的 MVC 编程 框架 ， 这 些 框 架 都 是 PHP MVC 中 的 典型 ， 借 助 
于 这 些 框架 的 设计 思想 ， 读 者 将 会 感受 到 PHP 编程 不 再 是 草根 语言 ， 而 是 一 门 高 级 优雅 的 
现代 化 编程 语言 。 

要 实现 MVC 开发 ， 最 简单 的 方式 就 是 使 用 现 有 成 熟 的 MVC 框架 ( 由 一 系列 PHP 类 及 
马 数 组 成 的 类 库 )， 在 开源 编程 世界 中 ， 遵 循 不 要 重复 造 轮子 的 道理 是 永远 相通 的 。 与 其 他 
语言 不 一 样 ， 在 PHP 中 实现 MVC 是 允许 继续 使 用 传统 的 函数 式 编程 ， 所 以 开发 人 员 可 以 方 
便 地 对 MVC 框架 进行 扩展 及 定制 。 但 正 是 由 于 这 些 特性 ， 寻 致 在 PHP 编程 世界 中 ， 并 没有 
统一 规范 的 MVC 框架 。 大 多 数 MVC 框架 都 有 各 自 一 套 规 范 。 

实现 PHP MVC 编程 的 框架 已 经 非常 多 ， 选 择 一 套 合适 的 MVC 框架 是 编程 人 员 首 先 需 
要 面 对 的 问题 。 本 章 将 会 介绍 最 流行 的 PHP MVC 框架 ， 并 演示 这 些 框架 的 简单 使 用 ， 通 过 
本 章 内 容 ， 直 观 地 认识 到 这 些 框架 的 功能 、 特 点 以 及 发 展 状 况 ， 为 后 续 自行 开发 MVC 框架 
提供 思路 。 

学 习 目 标 

@ 理解 Zend Framework。 
掌握 Zend Framework 的 部 芽 与 使 用 。 
掌握 Symfony 的 安 逆 与 使 用 。 
掌握 Symfony 控制 台 主 要 命令 的 使 用 。 
理解 CakePHP 实现 MVC 人 敏捷 开发 。 
掌握 CakePHP 的 实战 运用 。 
掌握 轻 量 级 CodeIsniter 框架 的 实战 应 用 。 
认识 国内 著名 的 ThinkPHP 框架 。 
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4.1 €B] Zend Framework 


Zend Framework 是 Zend 公司 针对 企业 级 开发 提供 的 一 套 MVC HER. CRE 
MVC 设计 思想 ， 提 供 组 件 化 的 编程 方式 ， 大 大 简化 了 PHP 的 开发 流程 。Zend Framework 是 
获得 最 多 商业 支持 的 PHP MVC HEX, HF Zend Framework DT Zend， 它 的 地 位 高 贯 、 
功能 强大 ， 上 所 以 Zend Framework 迅速 成 为 了 PHP 企业 级 开 友 的 自选 框架 。 


4.1.1 Zend Framework 简介 


Zend Framework 是 PHP MVC 设计 中 的 风 癌 标 ，Zend Framework 中 的 设计 思想 是 众多 
PHP MVC 框架 所 参照 的 样板 ， 因 为 Zend 公司 在 PHP 发 展 史 中 的 特殊 位 置 ， 所 以 Zend 
Framework 无 疑 是 PHP MVC 设计 中 的 重量 级 企业 级 产品 。 

事实 上 也 是 如 此 ， 包 括 IBM, Yahoo, Google 等 众多 商业 巨头 都 对 Zend Framework H 
开关 怀抱 ， 提 供 了 众多 商业 化 支持 ， 使 得 Zend Framework 的 模型 库 和 扩展 库 变 得 非常 强大 
和 易 用 。 典 型 的 有 Google API, Yahoo Mail, OAuth 等 。 

Zend Framework 是 基于 面 问 对 象 实现 的 ， 在 实现 MVC 的 层次 中 ， 社 区 开发 人 员 根 据 功 
能 将 Zend Framework 进行 组 件 化 ， 这 些 组 件 是 确保 Zend Framework 功能 强大 的 关键 ，PHP 
开发 人 员 使 用 Zend Framework 组 件 化 编程 ， 能 够 轻易 地 实现 高 效 编程 。 

在 Zend Studio 中 ， 己 经 对 Zend Framework 提供 了 支持 (如 直接 提供 项 目 创 建 、 代 人 码 管 
理 、 智 能 提示 、 代 但 跟踪 、 程 序 调试 等 )， 所 以 耽 算 是 初学 者 ， 也 能 够 迅速 上 手 。Zend 
Framework 没有 专门 的 视图 解释 引擎 ， 它 是 由 纯正 的 PHP 构成 的 ， 这 样 做 是 为 了 减少 程序 员 
的 学 习 成 本 ， 让 MVC 框架 回归 真正 的 PHP 编程 。 


4.1.2 SS Zend Framework 


这 里 介绍 的 Zend Framework 版 本 为 1.5.2， 个 人 可 以 免费 使 用 。 如 果 是 企业 ，Zend 公司 
提供 了 商业 化 支持 。 接 下 来 将 介绍 Zend Framework 的 安 状 与 使 用 。 

1. 安装 Zend Framework 开发 环境 

首先 ， 需 要 确保 运行 环境 能 够 满足 Zend Framework 的 需求 ， 这 里 将 采用 第 1 %31 TT 
绍 过 的 XAMPP 安装 包 进 行 搭建 。 正 常 启动 XAMPP 后 ， 输 入 http://localhost 将 能 够 看 到 配置 
成 功 画 面 ， 单 击 phpinfo 连接 ， 将 会 显示 PHP 的 运行 环境 ， 如 图 4-1 ra, 

由 于 Zend Framework 使 用 PDO 对 数据 库 进 行 操作 ， 搭 建 好 运行 环境 后 还 必须 手动 开局 
相应 的 PDO 功能 。 首 先进 入 xampp\php\ 目 录 ， 然 后 使 用 文本 工具 打开 php.ini 文件 ， 使 用 搜 
索 功 能 定位 到 所 有 pdo 模块 ， 然 后 将 相应 的 模块 注释 删除 ， 配 置 数 据 如 下 。 


extension=php_pdo.dl1 


extension=php_pdo_mysql.dl1 
extension=php_pdo pgsqgql.dll 
extension=php_pdo_sqgqlite.dll 
extension=php_pdo_odbc.dll 
extension=php_pdo_firebird.dll 
extension=php_pdo_oci8.dl1l 
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(2 XAMPP for Windows 7 


e RA 


XAMPP 


1-7.4 
[PHP: 5.3-5] 


System Windows NT WIN-J7H23PJANZY 6.1 build 7600 (Unknow Windows version 
Ultimate Ediion) 2286 


Spe [jon 62011 175045 


Compiler Je Vissal C++ 60) 
EE 
ees Configure jnologo configurejs -= ble-snapshot-build" *- -disable-isapi* == 
Instant Art BE hooia Es SE EE EE 
电话 第 "with-pdo-oci DAphp-sd 
pi “with-ccig=DAphp-sd edk, shared 
Perl cdb-lig=Diphp-sdkioracle\instantelienti 1\sdk shared" "~-enable-object 
parlinfoly cut-dir=. ob  --enable-com-dotnet *--with-meryptzstatic* 


| 


4-1 phpinfo 信息 


保存 php.ini 文件 ， 然 后 运行 xampp/apache_start.bat 程序 ， 重 新 局 动 apache 服务 ， 以 便 
配置 生效 。 完 成 后 束 能 够 进行 Zend Framework 程序 开发 了 。 


O 注意 : E Linx 环境 下 ， 相 应 的 模块 名 将 以 .so 为 后 缀 ， 例 如 extension = "pdo_mysql.so"。 在 Linux 环境 
T, PDO 功能 需要 开发 人 自行 安装 和 编译 ， 默 认 并 没有 安装 相应 的 pdo 模块 。 但 如 果 读 者 也 采用 Linux 
版 的 XAMPP， 该 套件 已 经 内 置 PDO 功能 。 


2. 部 署 Zend Framework 

Zend Framework 是 开放 源 代 人 码 的 ， 没 有 整合 到 PHP 类 库 中 ， 需 要 开发 人 员 目 行 下 载 并 
安装 。 打 开 http://www.zend.com/en/community/downloads 网 站 ， 找 到 需要 的 源 代 人 码 包 ， 下 载 
相应 的 Windows DL (HI zip 包 )， 然 后 解压 到 相应 的 目录 ， 将 目录 下 的 zend 文件 夹 复制 到 网 
站 的 根 目 录 下 (如 xampp/htdocs/zf/)， 整 个 安装 过 程 就 完成 了 。 

Zend Framework 是 由 一 些 常 见 的 PHP 文件 组 成 的 ， 理 论 上 只 需要 放 到 PHP 环境 下 就 可 
以 运行 。 但 是 由 于 Zend Framework 是 一 种 平台 ， 本 看 高 效 利用 的 原则 ， 可 以 将 Zend 
Framework 放 进 include_path 配置 中 ， 方 法 如 下 。 

打开 xampp/php/php.ini， 搜 索 ;include_path， 将 前 面 的 注释 删除 ， 重 新 启动 apache 服务 
器 ， 在 浏览 器 中 打开 phpinfo， 查 看 include_path 选项 ， 如 图 4-2 所 示 。 


4 XAM PP for Windows ning / Esp 


Ir Portugués (Brasil); Sg 


1-7.4 
[PHP 3.3.3] 


4-2 include_path 选项 
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如 图 中 所 示 ，include_path WJ H >K D:\Downloads\ixampp-win32-1.7.4-VC6\xampp\php\ 
PEAR， 只 需要 复制 或 移动 解压 出 来 的 zend 文件 夹 到 D:\Downloadsxampp-win32-1.7.4-VCA\ 
xampp\php\PEAR 目录 下 ， 束 可 以 实现 include_path Hé. SIE, Zend Framework 的 部 闭 束 完成 
了 ， 接 下 来 束 可 以 进行 Zend Framework MVC 开发 了 。 


O 说明: include_path 是 基于 pear 的 文件 处 理 机 制 ， 它 实现 了 PHP 的 环境 变量 功能 ， 能 够 让 PHP 引擎 在 
web 目录 下 找 不 到 相应 的 文件 时 自动 搜索 include_path 配置 目录 。include_path 支持 定义 多 个 目录 作为 
PHP 的 环境 变量 ， 定 义 多 个 目录 时 使 用 ; 陋 开 (Linux 使 用 : 隔 开 )。 在 PHP 代码 中 获取 include_path 目录 路 
径 时 ， 可 以 使 用 get_include_pathO žr. include_path 中 指定 的 目录 必须 肪 物理 路 径 ，Linux 系统 必须 要 
按照 正确 的 格式 填写 。 


4.1.3 使 用 Zend Framework 实现 MVC 


接 下 来 将 通过 Zend Framework 构建 一 个 简单 的 MVC 应 用 。Zend Framework 使 用 单一 
入 口 文件 ， 所 有 的 请 求 都 由 入 口 文件 进行 调配 。ZendFramework 的 核心 组 件 为 
Zend_Controller_Front， 在 入 口 文件 中 需要 对 该 组 件 进 行 初始 化 。 代 人 码 如 下 所 示 。 


<?php 


date default timezone set ('Asia/Shanghai'); 

require_once 'zend/Controller/Front.php'; 
Zend_Controller_Front::run('./application/controllers'); 
?> 


上 述 代 码 中 ， 首 先 使 用 PHP 内 置 国 数 date_default_timezone_set 设置 运行 环境 的 时 区 ， 这 不 
是 必需 的 ; 使 用 require_once 引入 ZendFramework 核心 组 件 〈 此 处 的 zend 目录 并 不 存在 于 当前 
网 站 下 )， 这 是 使 用 ZendFramework 的 前 担 ， 上 所 以 是 必需 步骤 。 最 后 使 用 Zend_Controller_Front 
静态 类 中 的 run 方法 初始 化 整个 MVC 架构 ， 这 时 就 可 以 使 用 ZendFramework 构建 网 站 应 
II, 

需要 注意 的 是 ， 在 run 方法 中 ， 指 定 了 一 个 参数 “./application/controllers”， 该 参数 是 必 
需 有 的 ， 它 声明 了 当前 入 口 文件 对 应 的 应 用 程序 目录 和 控制 器 目录 。application 指定 了 这 是 一 
个 应 用 ， 它 是 ZendFramework 约束 的 一 个 规范 ， 当 然 名 称 可 目 行 设置 的 ， 但 一 个 入 口 文件 必 
须要 对 应 一 个 应 用 。 应 用 是 MVC 开发 中 比较 重要 的 概念 ， 通 俗 来 讲 就 是 一 个 网 站 ， 当 然 也 
可 以 是 一 个 大 网 站 中 的 某 一 功能 模块 。 一 个 网 站 中 可 以 有 多 个 入 口 文 件 ， 但 入 口 文件 只 能 对 
应 一 个 应 用 。 通 常情 况 下 默认 的 入 口 文件 为 index.php。 


application 目录 代表 一 个 网 站 的 整体 ， 所 以 它 的 I controllers 一 一 控制 器 目录 
目录 结构 通常 都 是 与 网 站 相关 的 ， 比 如 网 站 的 公共 文 上 nes， ， 
件 、 运 行 日 志 等 。 典 型 的 目录 结果 如 图 4-3 所 示 。 | modes — >» MVC 模型 

上 述 目 录 中 ， 除 了 controllers, models, views 处 > OPS 一 一 一 一 = 脚本 文件 


| views MW C 视图 


其 他 目录 都 是 可 目 定 义 的 。 接 下 来 在 controllers 目录 
中 创建 一 个 默认 的 模型 ， 用 于 验证 ZendFramework 是 图 4-3 ZendFramework 项 目 目录 结构 
合 已 经 搭建 成 功 。 默 认 控 制 右 的 文件 名 为 IndexControllerphp， 代 但 如 下 所 示 。 


<?php 


require_once ("zend/Controller/Action.php"); 
include "zend/Date.php"; 
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include "zend/Version.php"; 
class indexController extends Zend Controller Action { 
public function indexAction(){ 
echo "当前 的 版 本 为 : ". Zend Version::VERSION; 
} 
public function userAction (){ 
echo "用 户 中 心 "; 
} 


} 


通过 浏览 器 访问 网 站 的 根 目 录 ， 如 果 正 确 显 示 ZendFramework 的 版 本 号 即 代 表 
ZendFrameork 安装 正确 ， 奋 则 需要 检查 include_path 目录 指 问 。 
上 述 代码 与 普通 的 PHP 类 代码 并 没有 区 别 ， 在 ZendFramework 平台 中 任何 普通 的 PHP 


类 只 要 继承 于 Controller 都 可 以 作为 MVC 中 的 模型 (文件 名 必须 亲 循 ZendFramework ap 
规范 )。 


4.1.4 Zend Framework 核心 组 件 


Zend Framework 是 以 组 件 的 方式 进行 代 但 构建 的 ， 组 件 的 概念 类 似 是 Java 中 包 的 概 
o Zend Framework 强大 之 处 束 在 于 提供 了 非常 多 的 实用 组 件 ， 开 友人 员 可 以 方便 地 使 
用 Zend Framework 提供 的 种 类 组 件 ， 轻 易 地 实现 原本 需要 复杂 设计 的 功能 。 第 用 的 组 件 
有 Zend_Validate、Zend_Filter、Zend_Cache、Zend_Mail、Zend_Db_Adapter 等 。 下 面 分 
别 介 绍 。 

1. Zend Validate 

Zend_Validate 组 件 用 于 表单 数据 的 校 验 。 通 常 在 提交 表单 时 为 了 数据 的 安全 和 完整 ， 除 
了 六 端 开发 人 员 需 要 对 数据 的 正确 性 和 完整 性 进行 校 验 外 ， 作 为 后 台 处 理 程序 也 应 该 进行 数 
据 校 验 。PHP 本 吴 束 提供 了 非常 多 的 校 验 手段 ， 例 如 正则 匹配 、 字 符 处 理 函 数 等 ， 但 都 需要 
烦琐 的 处 理 ， 而 且 编程 的 质量 直接 影响 到 数据 校 验 的 结果 。Zend_Validate 能 够 非常 智能 地 实 
现 数据 的 校 验 ， 开 发 人 员 不 需要 编程 复杂 的 代码 ， 只 需要 指定 需要 校 验 的 字段 和 检验 类 型 ， 
Zend_Validate 就 能 够 实现 高 效 及 安全 的 数据 验证 。Zend_Validate 组 件 包 类 库 如 图 4-4 所 示 。 


上 


Class Description 

zend captcha_Base Base class Tor Captcha adapters 
F | zend Yaldate Less Than 

d Zond_ Valdate_Alpha 


cand vadate GrasterThan 
Zen vadate Emaiëddress 
Zen vadate Alum 

A ef Ab P Flease note there ae two standalone test senpts for testing IDH characters 
Zend_validate_Hostname due to problems with file encoding 

CET Valdate_ Between 

zend Vaudate_lp 

Zend vadate Eie Ceunt Vaidator for counting all given files 


Zand valdate_ Flla_Extonsian validator for the fle extension of a fle 


RRRRERRRRRRR BR RER 


Zend vadate Pie Syre Vaigdator for the mavgnum size of a file up to a max of 2GR 

Zend_validate_Fila_Upload Yalidator for the maamum size of a file up to a mag of 268 

zend_Validate_File_MimeT ype Vahgäator for thë mm typ of a filg 

Zend_validate_File_Exists Vahdator which ehecks if the hle arasdy msta m the drectory 
nd Validate _File_ImageSize Validator for tha image size of a image file 

Zënd vandate Cecnum 


图 4-4 Zend_Validate 组 件 
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下 面 通 过 Zend Validate 中 的 Zend Validate EmailAddress 类 实现 对 Email 地 址 的 校 验 ， 


代码 如 下 所 示 。 
// 引 入 Zendq_Validate_EmailAddqress 的 类 文件 
require_once 'Zend/Validate/EmailAddress.php'; 
// 实 现 化 Zend_Validate_EmailAddress 类 


Şvalidator = new Zend Validate EmailAddress () ; 
if (S$SvalLiaqator->1isValiadq(Semai1l1)) { 
echo "电子 邮件 地 址 有 效 ; 
} else { 
// 输 出 错误 信息 
foreach ($validator->getMessages() as $messageID => $message) { 
echo "Validation failure 'ŞmessageID': Şmessage\n"; 
} 
} 
2. Zend Filter -Apa 
num. php 
Zend Filter 提供 了 完善 的 数据 过 渡 功 能 ， 能 够 将 一 些 存在 安 See, 
y 5 NA DASAL Am y N 2 BEE 
SS 3 Di 的 数据 在 插 入 数据 库 HIJ 进行 过 波 H tE N. o Zend_Filter 提供 Gë Dioirs php 
: SI DO E W Dir.ph 
的 Zend_Filter_Interface (SÆ) 内 置 了 一 套 完整 的 规则 ， 能 人 够 EI GE GE 
实现 在 国际 化 的 环境 下 处 理 一 些 诸如 货币 、 单 词 统计 等 实用 功 于 
fE. zé Zend_Filter 组 件 类 如 下 。 
nt.php 
> Zend Filter_Inflector: 变形 器 。 E intertace.php 


> Zend Filter_HtmlEntities: 人 处理 HTML. 

> addFilter: 连接 器 。 

Zend_Filter 组 件 所 包含 的 类 如 图 4-5 所 示 。 

以 下 代码 将 演示 Zend Filter HtmlEntities 过 滤 非 安全 HTML 
代码 ， 如 下 所 示 。 


<?php 
require_once 'Zend/Filter/HtmlEntities.php'; 
shtmlEntities = new Zend Filter HtmlEntities ()，; 
echo zene rile sgert e"; "Htrmolkntitiesii: 
echo Zen on lr EE EE EE 
eho Aene mi lreers s cer Ee ee 
EC Zn rl Me eole (Va SAS (< VA ona) 
EE EE Eer 
echo Zend Filter::get ('d:/www/framework/', 'Dir'); 


echo Zend Filter::get ('d:/www/framework/', 'BaseName'); 


D PregReplace.php 
Ø RealPath.php 

g StringToLower.php 
W StringToUpper.php 
D StringTrim.php 

J StripTags.php 


图 4-5 Zend_ Filter 组 件 


echa zend kFilter::get('6labc2 3430s**(()<>', "Int );,/ REES Svalue 


echo Zend Filter::get('67abc2 343^&*x'!，'StripNewlines');// 返 回 不 带 任 何 换 行 控制 符 的 


字符 串 


echo zenci EE Ee 


echo Zend_Filter::get ('ABCDefg', 'StringToLower'); // R E e ae BLW DU e E 


svalue。 abcdefg 


echo Zend Filter::get ('ABCDefg', 'StringToUpper');//ABCDEFG 


R 


3. Zend Cache 


Zend_Cache 能 够 实现 多 种 高 效 的 前 端 与 后 台 缓 存 功能 ， 包 括 文件 缓存 、 数 据 库 缓存 、 


Memcache 绥 存 等 。 其 中 Zend_Cache_Core 类 为 绥 存 组 件 的 核心 ， 它 提供 了 Zend 缓存 读 写 欣 
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制 、 绥 存 生命 周期 、 绥 存 序列 化 、 绥 存 方 式 等 。Zend_Cache 缓存 方式 如 下 。 

(1) 前 端 

> Core: 核心 缓存 ， 所 有 前 面 绥 存 类 都 必须 继承 于 Core- 

> File: 以 普通 的 文件 进行 缓存， 类 似 于 生成 静态 文件 。 

> Output: 捕获 并 绥 存 输出 ， 用 于 实现 页 面 局 部 缓存 。 

> Page: 缓存 页 面 ， 对 提高 效率 帮助 很 大 ， 因 为 一 旦 命中 缓存 ， 驳 直接 谈 取 组 存 并 输 
出 ， 不 再 执行 后 面 的 代码 。 支 持 以 session, cookie, get, post 作为 cache id 干扰 
僻 。 比 如 不 同 的 cookie 产生 不 同 的 绥 存 页 面 。 

> Class: 绥 存 静态 类 和 对 象 。 

> Function: AFRA. 

(2) 后 端 

> APC: 即 Alternative PHP Cache， 它 是 基于 Zend 引擎 的 一 个 第 三 方 缓存 扩展 ， 能 够 
绥 存 PHP 编译 后 的 代码 ， 达 到 类 似 于 Java、C# 中 的 预 编 译 效 果 。 

> File: 将 PHP 源 文件 以 表态 的 方式 进行 缓存 。 

> Memcached: 使 用 Memcached〈 一 个 文 持 分 布 式 架 构 的 内 存 数据 库 ) 缓存 数据 。 


> Sqlite : 著名 的 RX A IE Zr Hi JE r, 在 Linux、 |. Backend 
UNIX 上 广泛 地 被 使 用 ， 在 小 数据 的 情况 下 Sqlite |. Frontend 

的 性 能 是 非常 高 效 的 。 EE 
Gë Core pbp 


> Xcache: 功能 类 似 于 APC. 
Zend_Cache 组 存 组 件 位 于 在 Cache Hat, ÆRU 


Ø Exception.php 


图 4-6 所 示 。 图 4-6 Zend Cache 绥 存 组 件 
下 和 面 将 通过 代码 演示 Zend Cache 操作 Memcached， 代 人 码 如 下 所 示 。 
<?php 
set_include_path (get_include_path ().":/home/work/ZendFramework/"); 


require_once ("Zend/Loader.php"); 


spl_autoload register(array('Zend Loader', 'autoload')); 
$frontendOptions = array( 'lifeTime' => 30, // cache lifetime of 2 hours 
'automaticSerialization' => true, 'caching' => true); 
sbackendOptions = array ( 
'servers' => array (array ( 


HOSETH D 2 

voore” => E 

'persistent' => Zend_Cache_Backend_Memcached: :DEFAULT_PERSISTENT 
) ) ， 'compression'=>false); 


$cache = zend_Cache: :factory('Core', 'Memcached', sfrontendOptions, 
ŞbackendOptions); 


if (!Şcachestr = Ş$cache->load ('mypage')) { 
scachestr = USS cached EAr ETME E e 
echo $cachestr; 


Şcache->save ($cachestr, 'mypage'); 
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} elege { 
echo $cachestr; 


echo TT 
4. Zend Mai 
在 PHP 中 实现 发 送 邮 件 是 一 件 轻松 的 事 ，PHP 已 经 内 置 了 Mail 函数 ， 直 接 使 用 PHP 提 
供 的 Mail 函数 即 可 实现 高 效 的 邮件 发 送 。Zend_Mail 是 Mail P% Mail 
数 的 增强 版 ， 它 以 面向 对 象 的 方式 为 开发 人 员 提 供 强大 的 邮件 处 
理 功能 。 例 如 能 够 轻易 地 实现 邮件 附加 发 送 、SMTP 验证 发 送 、 E 
POP, IMOP 邮件 功能 等 。Zend_Mail 几乎 提供 了 处 理 邮 件 所 需 @ Exception.php 
要 的 类 ， 并 且 能 够 较 好 地 处 理 国际 化 编码 问题 。Zend_Mail 组 件 mechten 
文件 结构 如 图 4-7 所 示 。 Ø Storage.php 


Zend Mail 和 PHP 内 置 的 Mail 函数 一 样 ， 使 用 简单 、 广 
便 。 下 面 将 通过 代码 演示 Zend_Mail 的 使 用 ， 代 码 如 下 所 示 。 


<?php 
regurrte once T2end/Marl php"; 


图 4-7 Zend Mail 组 件 


require_once 'Zend/Mail/Transport/Smtp.php'; 
class logMail { 

private static $_config=array ('auth'=>'login', 
'username'=>'kfQ86055.com', 
'password'=>'123456'); 

private scacie SS mail = mwlily 

privyace staci $ cransgport = 257 

Sueolne EE SE EE (Sticle, Sooey)! 


Cey 1 

Sehi jJjlescdare (Ym) 

stransport = new Zend Mail_Transport_Smtp ('mail.yuyu.com',self::Ş$_config); 
Small = new Zend_Mail(); 


Şmail->setBodyText ($body); 
Şmail->setFrom('servicel86055.com'); 
Şmail->addTo ('kf@86055.com'); 
Smail l=->setSubjecre (Sticle, EE 
Şmail->send(Ş$transport); 
recúurcn Crue? 
}catch (Exception $e) { 
Şe->getTrace(); 
return false; 
} 
return false; 
} 
public static function logMail ($title, Sbody) { 
SEnlg—> Construct (Sticle, $bociy) 7 
} 
public function _ destruct () { 
} 
} 
new logMail(' 邮 件 标题 ', ' 这 是 一 封 测试 邮件 ')，; 
?> 
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5. Zend_Db Adapter Db 
Zend_Db_Adapter GER E 数据 JÆ 的 组 件 》 能 够 实现 |L Adapter 
简洁 高 效 的 数据 库 操作 。Zend_Db_Adapter 是 基于 PDO (一 1 Profiler 
个 面向 服务 的 数据 库 操作 套件 ) 的 ， 所 以 在 使 用 ag 
Zend Db Adapter 前 需要 确保 PHP 运行 环境 已 经 文 持 PDO ` Table 
模块 o g Exception.php 
SEEN 全 部 功能 ， 并 且 简 化 Ø Expr.php 
Zend_Db_Adapter 实现 了 PDO 全 部 功能 ， 并 且 简 化 了 Ø Profiler.php 
使 用 步骤 ， 使 得 Zend_Db_Adapter 蝎 加 适合 MVC 编程 。 Gë Select.php 
Zend_Db_Adapter 提供 的 CURD 快捷 方法 能 够 实现 快速 的 数 Ee 
据 创 建 、 更 新 、 读 取 、 删 除 等 冲 见 操作 。 文 件 结构 如 图 4-8 
所 示 。 图 4-8 Zend_Db_Adapter 组 件 
下 面 将 通过 代码 演示 Zend_Db_Adapter 连接 MySQL 数据 库 服 务 器 ， 并 执行 数据 查询 
的 功能 。 
<?php 
require_once 'Zend/Db.php'; 
Şparams = array ('host' SE 
'username' => 'root', 


'password' => 'root', 
'dbname' => 'ceiba'); 
// 驱 动 选择 PDO_MYSQL 
sdb = Zend_Db: :factory('PDO_MYSQL', Ş$params); 
Şstmt = $db->prepare('SELECT * FROM user WHERE user > :username'); 
// 使 用 bindvalue 绑 定 变量 
$stmt->bindValue ('username', 'ÆF W '); 
Şstmt->execute (); 
// 使 用 PDOStatement 对 象 Sresult 将 所 有 结果 数据 放 到 一 个 数组 中 
$rows = $stmt->fetchAll(); 
?> 


如 以 上 代码 所 示 ，Zend_Db_Adapter 在 使 用 方式 上 与 PDO_MYSQL 类 似 。 碍 询 数据 使 
用 fetchAll 方法 ;更 新 数据 使 用 update 方法 ; 删除 数据 使 用 delete。 不 同 的 只 是 声明 方式 不 
一 样 而 已 。 


4.2 ”功能 强大 的 Symfony 


4.2.1 Symfony 简介 


Symfony 中 文 名 称 为 “ 狐 磨 方 ”” 它 是 由 法 国人 开 友 的 一 套 开 源 的 PHP FRIE. d 
的 版 本 于 2005 年 发 布 ，Symfony 定位 非常 明确 ， 它 整合 了 PHP5 所 有 功能 特点 ， 针 对 大 型 
应 用 特别 是 扩展 型 的 应 用 提供 了 整套 解决 方案 。Symfony 的 强大 之 处 在 于 它 的 Project ORM 
数据 库 模 型 ， 还 有 借鉴 于 Ruby on Rails 的 模板 处 理 引 警 以 及 分 工 明 确 的 Model-View- 
Controlle 设计 模式 。 

Symfony 在 国外 应 用 得 非常 广泛 ， 如 Digg, Engadget 等 都 在 使 用 。 而 在 国内 Symfony 
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比较 少见 ， 不 是 因为 Symfony 不 够 优秀 ， 也 不 是 Symfony 不 能 胜任 开发 需求 ， 而 是 Symfony 
在 国内 的 普及 还 处 于 初级 阶段 ， 最 需要 的 开发 文档 大 都 处 于 基文 或 天 文 翻译 的 阶段 。 本 市 将 
简单 介绍 Symfony 实现 MVC 开发 的 过 程 ， 了 解 Symfony 对 敏捷 开发 的 支持 。 


4.2.2 ”获得 Symfony 


要 使 用 Symfony 开发 Web WH, PHP 环境 必须 为 5.0.1 以 上 ， 并 且 安 装 了 Pear 扩展 
库 。 如 果 谈 者 和 笔者 一 样 使 用 XAMPP 套件 作为 开发 环境 ， 那 么 所 需要 的 环境 是 足够 的 。 接 
K PHA Z Symfony DIAS (CLA Windows 7 为 例 )。 

首先 将 XAMPP 下 的 php CD D:\Downloads\ixampp-win32-1.7.4-VC6\xampp\php) 目录 加 
入 操作 系统 的 环境 变量 ， 方 便 使 用 pear 命令 ， 如 图 4-9 所 示 。 


编辑 用 户 变 星 .se a we e 
变量 名 (有 ): Path 
RIY): 


图 4-9 将 php 目录 加 入 环境 变量 


然后 升级 PHP 日 之 的 pear Æ, MEU Fo 


pear upgrade pear 
HÆ a LAIRE Symfony 了 。 使 用 pear remote-list -c symfony 命令 列 出 可 用 的 包 ， 
如 图 4-10 所 示 。 


C: \Wsers\Administrator>pear remote-list -c symfony 


图 4-10 symfony 频道 可 用 的 包 


如 图 中 所 示 ， 当 前 的 Symfony 最 新 的 版 本 为 1.4.17， 当 然 这 指 的 是 正式 包 ， 实 际 上 可 用 
的 2.0 测试 包 已 经 可 供 下载 了 。 

接 下 来 将 进入 Symfony 的 安装 阶段 ， 请 确保 当前 网 络 环境 稳定 可 靠 。 此 过 程 将 会 视 网 络 
状况 ， 下 载 进 度 有 所 不 同 。 在 命令 行 中 输入 pear install symfony/symphony 指令 ，pear 将 会 把 
Symfony 安装 到 php 扩展 目录 下 ， 如 图 4-11 所 示 。 


C: \Users\Administrator>pear install symfony/symfony 
downloading symfony-1.4.17.t9z ... 
$starting to download symfony=-1.4.17.toz (3,187,861 bytes) 


install ok: channel://pear .Symfony-project,com/symfony-1 .4.17 


图 4-11 pear Zł% Symfony 
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TZR TC Ji n] LIE xampp\php\pear 目录 下 看 到 Symfony XFX, wk 4-12 Dro, 


几 MIME |. PHPUnitz 内 XML 

L MP3 |. Services Oepdb 

IL Net 几 SOAP .depdblock 
lL Numbers 几 Spreadsheet filemap 

I OLE I SQL dock 

Los |. Structures Auth.php 

|L Pager l symfony | F Cache.php 

L PEAR l. System J class.erpdf.pt 
IL phing L Testing g class.pdf.php 
|L PHP L Text J Config.php 

内 PHPDoc l Translation? g Contact Vear 
|L PhpDocumentor JL Var_Dump g Contact Vecar 
[L PHPUmIt 内 VersionControl Q Date.php 

1 


图 4-12 Symfony 安装 成 功 


测试 Symfony 是 否 可 用 ， 最 直接 的 办 法 就 是 创建 一 个 项 目 。 接 下 来 将 创建 一 个 Helo 项 
上 月， 体验 Symfon 开发 MVC 应 用 的 过 程 。 


4.2.3 ”实现 一 个 简单 的 MVC 


使 用 pear FREZI Symfony 后 ， 会 将 Symfony 包 的 脚本 命令 一 并 安装 ， 这 也 是 
Symfony 的 一 大 特色 。 在 命令 行 中 使 用 symfony -V 命令 将 会 显示 Symfony 当前 的 版 本 。 
symfony 命令 提供 了 完善 的 项 目 管 理 操作 ， 例 如 项 目 创 建 、 单 元 测试 、 调 试 以 及 Symfony 框 
架 本 喘 的 升级 维护 等 。 作 为 开发 人 员 ， 使 用 最 多 的 还 是 利用 symfony 命令 创建 项 目 及 应 用 ， 
下 面 我 们 束 使 用 symfony 命令 创建 一 个 测试 项 目 。 

1. 创建 项 目 

在 网 站 根 目 录 下 创建 一 个 文件 来， 并 命令 名 为 hello， 该 目录 用 于 存放 项 目 文件 。 打 开 
命令 行 ， 定 位 到 hello 项 目下 ， 接 下 来 的 操作 都 位 于 该 目录 中 。 使 用 symfony generate:project 
helo 命令 创建 一 个 名 为 helo 的 项 目 ， 该 项 目 是 基于 Symfony 框架 的 ， 命 令 执行 完毕 后 ， 
symfony 工具 上 自动 创建 了 所 需要 的 项 目 文件 和 目录 ， 如 网 4-13 所 示 。 


用 apps 项 目 应 用 文件 

用 cache 项 目 缓存 

用 config 项 目 配置 文件 

L data 数 年 库 ORM 映射 文件 
KL lib 项 目 增强 类 库 

log 项 目 日 志文 件 


IL plugins 一 一 一 一 一 一 一 一 Symfony 扩展 


test PHP Unit 单元 测试 
J web 对 外 发 布 文件 目录 
_] symfony 项 目 全 局 声明 文件 


图 4-13 项 目 目录 结构 


如 图 中 所 示 ， 就 是 一 个 典型 的 Symfony 项 目 目录 ， 其 中 的 web 目录 为 网 站 《 即 应 用 ) 
根 目 录 ， 创 建 完 项 目 后 还 需要 创建 应 用 ， 在 Symfony HERPA run, mm 
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目 只 是 一 个 便于 管理 的 模式 ， 应 用 才 是 具体 的 行为 ， 一 个 项 目 最 少 需 要 创建 一 个 应 用 (最 多 
可 以 无 限 )。 

2. 创建 应 用 

应 用 存放 于 ap 目录 下 ， 一 个 项 目 可 以 有 多 个 应 用 。 比 如 前 台 应 用 可 以 命名 为 home:; 
吾 台 应 用 命名 为 admin， 在 团队 开发 时 就 可 以 非常 方便 地 进行 管理 。 

>symfony generate:app home 

上 述 命 令 运 行 后 将 创建 一 个 名 为 home 的 应 用 。 结 构 如 图 4-14 所 示 。 

其 中 config 目录 存放 应 用 配置 信息 文件 ， 例 如 路 由 配置 、 绥 存 配 置 、 数 据 库 信息 配置 
E; lib 目录 存放 应 用 核心 类 库 ，'，modules 目录 存放 控制 嚣 ;，templates 目录 存放 全 局 布局 文 
件 ， 从 这 就 可 看 出 Symfony 能 够 很 好 地 管理 着 项 目 与 应 用 之 间 的 对 应 关系 ， 并 旦 提供 了 非常 
清晰 及 合理 的 MVC 设计 模式 。home 目录 里 的 文件 都 是 后 台 的 ， 只 需要 被 开发 人 员 看 到 ， 而 
不 需要 前 台 用 户 访问 。home 的 前 台 文 件 已 经 被 生成 到 了 we 目录 里 ， 事 实 上 we 目录 只 需 
要 存放 应 用 的 入 口 文件 和 应 用 的 公共 资源 目录 即 可 ， 如 图 4-15 所 示 。 


L css 
l images 
L js 
l uploads 
l. config Ø home_dev.php 
|, i18n 局 .htaccess 
bb lib index.php 
modules ] robots.txt 
|. templates Ø info.php 
图 4-14 home 应 用 目录 结构 图 4-15 web 目录 结构 


如 图 中 所 示 ，web 目录 存放 允许 被 用 户 访问 的 资源 以 及 文件 ， 由 于 home 项 目 为 第 一 个 
创建 的 应 用 ， 所 以 symfony 命令 工具 会 自动 地 将 该 应 用 设置 为 项 目的 主 应 用 ( 即 入 口 应 
用 ) ， 主 应 用 的 入 口 文 件 为 index.php， 如 果 再 创建 一 个 名 为 demo 的 项 目 ， 那 么 项 目的 入 口 
文件 就 会 变 为 demo.php。 

home_dev.php 是 Symfony 特有 的 项 目 调 试 文件 。 访问 index.php 文件 并 不 会 看 到 调试 中 
的 异常 信息 ， 但 访问 home_dev.php 文件 可 以 方便 地 查看 到 所 有 应 用 调试 中 的 信息 。 
XXX_dev.php 只 能 用 于 调试 阶段 ， 一 旦 正式 发 布 应 用 ， 应 该 要 将 其 删除 。 

经 过 前 面 的 介绍 ， 现 在 我 们 已 经 非常 地 清楚 了 Symfony 处 理 项 目 与 应 用 文件 结构 的 特 
性 ， 虽 然 symfony 命令 工具 生成 了 很 多 层次 的 目录 结构 ， 但 归根 结 故 用 户 访 问 到 的 只 有 web 
目录 ， 所 以 只 需要 将 域名 绑 定 到 项 目下 的 web 目录 ， 即 可 完成 项 目 部 署 。 

3. WERZA 

N TRMA KEANE TA, F FEK: hello.localhost 域名 绑 定 到 web 目录 。 
打开 CAWINDOWS\system32\drivers\etc\hosts 文件 ， 在 文件 中 添加 上 hello.localhost 域名 ， 内 


容 如 下 。 
# T27 OO localhost 
1127-0-01 Localhost 
127.0,.0.1 hello, localhost 
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打开 apache 配置 文件 (xampp 的 虚拟 主机 配置 文件 为 xampp\apache\conf\extra\httpd- 
vhosts.conf)， 将 hello 项 目下 的 web 目录 作为 一 个 虚拟 网 站 添加 到 配置 文件 中 。 


<VirtualHost *:80> 
ServerAdmin webmaster@localhost 
DocumentRoot "D: \xampp\htdocs\symfony\hello\web" 
ServerName hello.localhost 

</VirtualHost> 


上 述 配置 操作 都 是 基于 Windows 7 的 ， 如 果 使 用 的 是 Linux 系统 ， 大 体 的 步骤 是 一 样 
的 ， 只 是 在 配置 本 地 域名 时 使 用 hostname 命令 。 配 置 完成 后 重新 局 动 apache， 以 使 新 网 站 
生效 ， 此 时 通过 http://helo.localhost 域名 应 能 够 正确 地 访问 到 home 项 目 中 的 web 目录 。 

4. 创建 模块 

经 过 前 面 的 步骤 ， 现 在 已 经 创建 了 1 个 项 目 、1 个 应 用 。 在 Symonfy 框架 中 上 述 步骤 只 
是 创建 一 个 网 站 的 前 提 ， 在 Symonfy 中 还 有 一 个 非常 重要 的 概念 ， 那 瓯 是 模块 。 在 前 一 和 介 
绍 的 Zend Framework 框架 中 ， 虽 然 也 有 模块 ， 但 概念 并 不 是 很 清晰 ， 通 常 称 之 为 控制 堪 ， 
而 在 Symonfy 中 模块 的 概念 非常 清晰 ， 筷 更 像 一 个 网 站 中 的 子 网 站 或 者 频道 ， 它 的 模块 由 一 
系列 的 控制 句 文 件 构 成 ， 默 认 的 控制 器 文件 为 actions.class.php。 和 下面 将 使 用 generate:module 
命令 创建 一 个 名 为 news 的 模块 。 

>symfony generate:module home news 

上 述 命 令 将 会 在 home 应 用 下 创建 一 个 名 称 为 news 的 模块 ， 打 开 appsihome\modules\ H 
录 ， 将 会 看 到 由 symfony 工具 生成 的 news 模块 ， 并 且 生 成 了 actions 目录 及 templates 目录 。 
actions 用 于 存放 模块 控制 器 文件 ， 上 默认 生成 的 actions.class.php 控制 器 文件 代码 如 下 。 


<?php 


/** 


* news actions. 
GPackade hello 


大 
大 
* Qsubpackage news 
x Qauthor Your name here 
x @version SVN: $Id: actions.class.php 23810 2009-11-12 11:07:44Z Kris.Wallsmith $ 
E 
class newsActions extends sfActions 
Í 
/大 大 
x Executes index action 
大 


* Qparam sfRequest Srequest A request object 
EA 
public function executeIndex (sfWebRequest Sreduest ) 
{ 
$this->forward('default', 'module'); 


} 
} 
templates 存放 控制 器 动作 相对 应 的 模板 ， 默 认 已 生成 indexSuccess.php 文件 ， 该 文件 即 
为 index 动作 对 应 的 模板 。 
5. 开发 应 用 
仍然 以 前 面 创建 的 hello.localhost 应 用 为 例 ， 默 认 创 建 的 news 模块 有 一 个 默认 的 动作 ， 
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HI index 动作 。 打 开 apps/home/modules/news/actions/actions.class.php 文件 ， 接 下 来 添加 一 个 
名 为 hello 的 动作 ， 在 hello 动作 中 输出 一 串 字 符 ， 代 但 如 下 所 示 。 


public function executeHello (sfWebRequest S$request) 
{ 
$info=" 欢 迎 使 用 Symfony"; 
sthis—>info=$info; 


} 
上 述 代码 是 一 个 名 为 helo 的 控制 右 动 作 ， 可 以 看 到 它 的 命名 方式 为 execute+ 动 作 名 。 
其 中 动作 名 称 的 首 字 母 必须 大 写 。 通 过 http://hello.localhost/news/hello 将 能 够 访问 到 该 动作 ， 
在 Symfony 框架 中 ， 动 作 与 模板 必须 要 成 对 出 现 ， 这 意味 看 还 必须 要 创建 一 个 对 应 的 模板 文 
件 才 能 正常 运行 。 接 下 来 在 apps/home/modules/news/templates/ 目 录 中 创建 动作 模板 ， 并 命名 
为 helloSuccess.php， 人 代码 如 下 所 示 。 


<html> 


<head> 


<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
wcle m Ee 

</head> 

<body> 

<?php echo $info;?> 

</body> 

</html> 


AEE, ERA IE V hlhttp:/hello.localhost/news/hello T - 


4.2.4 Symfony 的 配置 文件 


配置 文件 在 Symfony 中 具有 非常 重要 的 作用 。Symfony 使 用 Yam 文件 作为 配置 文件 。 
Yaml 是 一 种 徐 单 紧凑 的 数据 序列 文件 ， 它 不 像 XML 那样 需要 标签 ， 也 不 像 JSON 那样 需要 
格式 配对 ， 它 使 用 空格 作为 标记 ， 这 些 空格 能 够 被 Yaml 解释 器 关 认 ， 然 后 转换 成 PHP 数组 
代码 。 以 下 代码 束 是 一 个 简单 的 Symfony 配置 文件 ， 代 码 如 下 。 


# You can find more information about this file on the symfony website: 


# http://www.symfony-project.org/reference/1_4/en/09-Cache 


default: 
enabled: false 
with_layout: false 
lifetime: 86400 


如 以 上 代码 所 示 ， 定 义 了 home 应 用 的 绥 存 东 上 略 ， 如 来 要 访问 配置 文件 里 的 配置 信息 ， 
可 以 使 用 sfConfig::sgetO 方 法 获取 。 


4.3 ”灵活 完善 的 CakePHP 


4.3.1 CakePHP 简介 
CakePHP 是 另 一 款 近 年 来 广 受 好 评 的 PHP 应 用 开发 框架 ， 多 年 来 一 直 被 众多 国外 开 
E Wi 
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源 社区 评 为 最 优秀 的 PHP 应 用 开发 框架 之 一 。CakePHP 不 需要 设置 PHP Path 也 能 够 部 
普 在 任意 层次 的 网 站 目录 结构 中 ， 它 的 视图 引擎 文 持 多 种 语法 ， 不 仅 文 持 原生 的 PHP 语 
法 ， 还 支持 syntax 等 语法 。CakePHP 在 PHP 4 时 代 就 已 经 非常 有 名 气 ， 在 PHP 5 中 继承 
了 早期 版 多 数 优点 ， 并 提供 了 众多 新 特性 。 其 中 稳定 与 灵活 是 新 版 CakePHP 最 显著 的 特 
Ee 

CakePHP 由 一 个 PHP 源 代码 包 所 组 成 ， 它 使 用 科学 、 合 理 的 层次 结构 将 CakePHP 分 为 
数据 库 、 应 用 程序 脚手架 、 代 码 产 生 器 、URL 人 处理、 数据 验证 、 模 板 引 擎 、ACL 权限 、 绥 
存 处 理 、 国 际 〈 本 地 ) 化 等 十 多 个 模块 。CakePHP 的 代码 量 非常 小 ， 读 者 可 以 
http://cakephp.org/ 下 载 到 CakePHP 的 最 新 版 本 。 接 下 来 将 使 用 CakePHP 构建 一 个 简单 的 
MVC 应 用 ， 让 读者 对 CakePHP 有 一 个 直观 的 认识 。 


4.3.2 Fass CakePHP 


打开 http:/cakephp.org/ 网 站 ， 单 击 导 航 栏 中 的 “Downloads ”链接 ， 选 择 相 应 的 
CakePHP 版 本 ， 这 里 选择 2.0。 然 后 将 进入 GitHub 网 站 ， 并 选择 相应 的 下 载 包 ， 解 压 后 将 会 
得 到 CakePHP 开 友 框 染 源 代 人 码 ， 代 公 如 图 4-16 所 示 。 


"a "mm "i = ` 
ALE | a 
app lib plugins vendors .gitignore „htaccess 

.travis.ymil build.prop build.xml index.php README 


erties 
图 4-16 CakePHP 框架 目录 结构 


如 图 中 所 示 ，app 目录 存放 着 MVC 应 用 所 需要 的 文件 、lib H TEEN 
KIFER KOX F plugins 目录 存放 内 置 扩展 文件 、vendors Console 


目录 存放 第 三 方 扩展 文件 。 
对 于 开发 者 而 言 主要 需要 处 理 app 目录 ， 该 目录 中 有 一 个 

webroot 目录 ， 它 束 是 整个 网 站 的 入 口 ， 所 有 其 于 CatePHP 构建 Ñ Plugin 

的 网 站 都 由 该 目录 引导 进入 ， 用 户 访 问 到 的 路 径 始终 都 是 该 目 

录 。 所 以 无 论 是 真实 的 生产 环境 还 是 开发 环境 ， 为 了 便于 测试 都 L Vendor 

应 该 将 域名 绑 定 到 该 目录 《〈 绑 定 方法 可 参照 4.3.3 节 )， 这 里 绑 定 

的 域名 为 catephp.localhost。app 目录 结构 如 图 4-17 所 示 。 3 .htaccess 

Vë index.php 


4.3.3 ”使 用 CakePHP 构建 MVC 编程 


图 4-17 app 目录 结构 


在 CakePHP 中 ， 模 型 Modue 与 控制 器 (Controller) 是 非常 依赖 的 一 对 关系 ， 模 型 又 与 
后 人 台 的 数据 表 一 一 对 应 〈 通 名 情况 下 一 个 模型 影射 一 个 数据 表 )。 与 其 他 框架 不 一 样 ， 
CakePHP 不 允许 使 用 空 视 图 。 下 和 面 通过 一 个 人 简单 的 示例 ， 演 示 CakePHP 创建 一 个 简单 的 


MVC 网 站 的 过 程 。 
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1. 配置 效 据 库 连 接 

CakePHP 的 配置 信息 存放 于 app/Config 目录 ， 这 些 配置 文件 包括 框架 核心 配置 、URL 
路 由 配置 、ACL 权限 配置 、 邮 件 系 统 配置 等 。 配 置 文件 系统 并 不 会 目 动 生成 ， 但 默认 生成 了 
后 级 名 为 .default 的 配置 示例 。 这 里 只 需要 对 数据 库 进 行 配置 即 可 。 

打开 database.php.default 文件 ， 将 该 文件 男 存 为 database.php， 修 改 其 中 的 数据 库 配 置信 
忆 ， 代 码 如 下 所 示 。 


<?php 
class DATABASE_CONFIG { 
public sdefault = array ( 


'datasource' => 'Database/Mysql', 
e EE DEE EE 
tee 

E EE EE 

'password' => 'root', 

'database' => 'cakephp', 

EE > 


emeoenme => VUES p 
} 


> 
2. 创建 数据 库 
前 面 提 到 过 CakePHP 的 模型 对 应 一 个 数据 表 ， 此 处 将 创建 一 个 名 为 Users 的 数据 表 ， 需 
要 注意 的 是 在 CakePHP 里 通常 表 名 都 是 以 s 结尾 的 ， 这 是 为 了 与 模型 更 好 地 整合 而 采用 的 
一 种 特殊 的 命名 方式 ， 当 然 这 只 是 个 约定 ， 并 非 必 需 的 。SQL 语句 如 下 。 
CREATE TABLE 'Users' ( 
'id' tinyint (4) NOT NULL AUTO _ INCREMENT, 
'user_name' CHAR( 20 ) NOT NULL , 


'user_email' CHAR( 40 ) NOT NULL , 


INDEX ('user_name' , 'user_email' ) 


) 
REN User KIERA AA TTR o 
INSERT INTO 'Users' ( 
'user_name' , 
'user_email' 
) 
VALUES ( 
' 李 开 涌 :， 'kf@86055.com' 
RAR 


'ceiba', 'ceiba_Q126.com' 


); 

3. 创建 数据 表 模 型 

接 下 来 需要 为 User 数据 库 表 创 建 一 个 对 应 的 数据 模型 。CakePHP 的 模型 文件 存放 于 
/app/models 目录 下 ， 此 处 需要 创建 的 模型 文件 名 为 Users.php， 代 人 码 如 下 。 


<?php 
//User 模型 
class User extends AppModel { 


var $name = 'User'; 


} 


?> 
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由 于 数据 表 名 为 Users， 所 以 模型 的 文件 名 称 也 必须 为 Users.php。 通 过 这 些 受 约束 的 文 
件 命 名 方式 ， 开 发 人 员 甚 至 不 需要 写 任何 代码 ， 束 可 以 让 数据 表 与 数据 模型 产生 关联 。 

4. 创建 控制 器 

模型 (Model)〉 只 能 锐 调 用 ， 而 不 能 直接 被 访问 。 接 下 来 需要 创建 一 个 控制 占用 于 让 用 
户 访 问 ， 并 将 Users 表 中 的 数据 显示 到 界面 上 。 控 制 占 存放 于 /app/controllers 目录 里 ， 这 里 
将 创建 名 为 UsersController.php 的 控制 嚣 文件， 代码 如 下 。 


<?php 
class UsersController extends AppController { 


public $name = 'Users'; 
function index() { 
$this->set ('Users', Ş$this->User->find('all')); 


} 
} 


R 


如 上 述 代 人 码 所 示 ， 其 中 UsersController HPEH 23 gr. index 为 动作 名 【不 区 分 大 小 
$this->set 表示 将 合 询 结果 分 配 到 变量 Users， 访 变量 将 在 View 中 使 用 。 

5. 创建 视图 

前 面 的 步骤 完成 后 ， 可 以 通过 http://cakephp.localhost/users/index 访 问 到 UsersController 

控制 器 下 的 index 动作 ， 但 是 由 于 还 没有 创建 视图 模板 ， 所 以 CakePHP 将 报错 。 视 图 模板 文 

件 存 放 于 app/View 目录 下 ， 这 里 首先 需要 创建 名 为 Users 的 视图 模板 目录 (与 Users 控制 器 

对 应 )， 然 后 创建 一 个 index.ctp 视图 文件 (与 index 动作 对 应 )， 代 码 如 下 所 示 。 


<table> 
<?php foreach ($Users as $user): 
KERS 
<td> 
<?php echo Ş$user['User'] ['user_name']; 
</td><td> 
<?php echo Ş$user['User'] ['user_email']; 
</td> 
</ EES 
<?php endforeach; 
</table> 


此 时 通过 http://cakephp.localhost/users/index 网 址 访问 ， 将 会 循环 显示 Users 表 中 的 数据 ， 
效果 如 图 4-18 Dirr 


Di 


?> 


> 


R 


?> 


(e 


<| SS CakeP... x 
< ™ U G c 


r Query 


1 SELECT 
"User" 


kf@86055.com 


ceiba_@126.com 


Error Affected 


"User, id ， 2 
"user Dame - 

er user email FROM 
`~. users ` A r 


S "User" WHERE 


Num. Took 
rows (ms) 


2 2 


8 On 


index 动作 视图 效果 
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需要 注意 的 是 ， 在 CakePHP 中 布局 样式 决定 了 整个 网 站 的 风格 ， 文 件 位 于 app/View/ 
Layouts 目录 下 ， 设 计 人 员 需 要 改变 布局 文件 才能 最 终 改变 动作 视图 的 外 观 效果 。 


4.3.4 WABI CakePHP 视图 助手 


CakePHP 有 一 个 好 用 的 视图 助手 ， 它 能 够 以 组 件 化 的 方式 为 视图 文件 快速 添加 布局 元 
素 ， 有 效 地 减少 HTML 代码 的 编写， 提高 开发 速度 。2.x 版 本 的 视图 助手 还 能 够 提供 页 面 局 
部 绥 存 ，AJAX、 样 式 目 定义 等 实用 功能 。 接 下 来 将 通过 示例 代码 演 示 CakePHP 视图 助手 的 
使 用 。 

打开 app/View/Users/index.ctp 视图 文件 ， 将 该 文件 的 循环 代码 改 成 以 下 代码 。 


<?php foreach (SUsers as Suser): ?> 
Ee 

< 七 Q> 

<?php echo $this->Html-—->link ($user['User']['user name'], 
array ('controller' => 'Users', 'action' => 'view', $user['User']['id'])); ?> 

</td> 
<td> 

<2pao eche SVS Useri user email Yl; ?3 


</td> LTES 
<?php endforeach; ?> 


如 以 上 代码 所 示 ， 其 中 this->Html->linkO 残 古 视图 助手 方法 ， 该 方法 派生 于 Htm 处 理 
类 ， 该 类 包含 了 律 用 的 视图 助手 方法 ，link 方法 表示 一 个 超 链 接 组 件 ， 并 接受 2 个 参数 。 其 
中 参数 1 将 用 于 生成 超 链 接 字 符 ， 参 数 2 将 定义 链接 参数 。 除 了 link 助手 外 ，HTML 其 类 未 
提供 了 非常 多 的 实用 组 件 ， 例 如 图 厂 助 手 ， 如 以 下 代码 所 示 。 


<div id="footer"> 
<?php echo $this->Html->link( 
EIERE (V Cake Power ol arcay (Vale) => 


ŞcakeDescription, 'border' E 
'http://www.cakephp.org/', 
array ('target' => '_blank', 'escape' => false) 


EK 
</div> 


从 以 上 代码 可 以 看 出 ，CakePHP 框架 的 视图 助手 可 以 方便 地 生成 视图 布局 元 素 。 以 上 代 
码 虽 不 能 体现 出 视 网 助 手 的 便捷 性 ， 但 如 条 在 表单 、 表 单 域 、AJAX 等 需要 大 量 脚 本 以 及 
HTML 的 场合 ， 视 图 助手 确实 能 够 简化 许多 脚本 代码 。 由 于 视图 助手 使 用 纯正 的 PHP 代替 
HTML 及 JavaScript， 所 以 View 页 面 看 上 去 也 更 加 简洁 。 


4.4 使 用 广泛 的 Codelgniter 


4.4.1 Codelgniter 简介 


Codelgniter 人 简称 CI， 是 国内 使 用 最 广泛 的 PHP MVC 框架 之 一 。Codelgniter z KIE 
Apache/BSD 双 协 议 的 开源 产品 ， 所 以 开发 人 员 可 以 将 CodeIgniter 整合 到 任何 开发 环境 ， 不 
需要 担心 版 权 问 题 。 
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CodeIgniter 从 MVC AU) Val CT, SS rom. Codelgniter 定位 于 
小 型 的 网 站 应 用 开发 ， 但 并 不 代表 Codelgniter 不 适用 于 大 型 网 站 开发 。 事 实 上 ，CodeIsniter 
对 MVC 的 文 持 是 非 铝 强大， 无 论 是 URL 的 处 理 ， 还 是 视图 引 苟 都 能 够 适应 大 型 网 站 的 开 
发 需求 。CodeIsniter MVC 处 理 流 程 如 图 4-19 PZR. 


一 
DS =e 

=r — [ra | — SES 

Index.php | om SES 

e 一 [更 了] 一 Sp 

Ss 


图 4-19 CodeIgniter MVC 处 理 流 程 


如 图 中 所 示 ，CodelIgniter 在 URL 处 理 阶 段 就 提供 了 安全 检验 ， 对 GTE 和 POST gr T 
PHP 内 置 的 处 理 方 式 ， 提 供 了 完善 的 过 滤 方 法 。 控 制 噩 负责 调用 模型 、 类 库 、 辅 助 函 数 、 插 
件 、 脚 本 等 功能 模块 ， 其 中 模型 通 种 就 是 处 理 数 据 库 表 的 实体 影射 ， 它 是 整个 MVC 的 桥 
染 。 辅 助 函 数 是 CodeIgniter 框架 的 一 个 实用 而 强大 的 功能 库 ， 许 多 原本 复杂 的 功能 调用 辅助 
函数 就 能 够 轻松 地 完成 。CodeIgniter 还 文 持 官 方 及 第 三 方 的 扩展 ，PHP 程序 员 使 用 面 癌 对 象 
知识 可 以 快速 地 开发 出 CodeIgniter 扩展 。 

CodeIgniter 文 持 主流 的 缓存 机 制 〈 如 文件 缓存 、 内 存 缓存 、 数 据 库 缓存 、 浏 览 器 压缩 组 
存 等 )， 并 且 文 持 页 面 局 部 缓存 、 控 制 右 缓存 、 数 据 库 绥 存 等 ， 这 些 缓存 有 些 是 智能 化 的 ， 
有 些 需 要 开发 人 员 目 行 处 理 ，CodeIsniter ZU T 3E Odre CA, 

单元 测试 是 应 用 程序 开发 中 重要 的 一 个 环节 ，Codelgniter 支持 主流 的 PHP 单元 测试 ， 
它 是 确保 应 用 质量 的 关键 手段 ， 不 地 的 是 PHP 程序 员 并 不 习惯 使 用 单元 测试 ，CodeIsniter 
提供 了 人 徐 单 但 强大 的 单元 测试 ， 有 效 地 降低 了 单元 测试 难度 ， 从 而 提高 应 用 的 代码 质量 。 不 
仅 如 此 ，CodeIgniter 还 提供 了 数据 库 脚 手 染 ， 开 发 人 员 在 不 进入 数据 库 系 统 的 情况 下 束 可 对 
数据 进行 整理 、 测 试 等 ， 从 而 提高 PHP 应 用 开发 速度 。 

CodeIgniter 灵活 高 效 的 MVC 处 理 机 制 ， 无 论 在 国内 还 是 国外 都 得 到 了 广泛 的 PHP 程 
序 员 文 持 。 在 开源 社区 还 衍生 了 Kohan 子 项 目 。 对 于 国内 PHP 程序 员 来 说，CodeIgniter 
不 仅 简单 好 用 ， 而 且 官 方 提 供 了 详细 的 中 文 攻 助 文档 ， 这 也 是 CodeIgniter 在 国内 流行 的 原 
因 之 一 。 

接 下 来 将 以 Codelgniter 2.1.2 为 基础 ， 介 绍 Codelgniter 的 MVC 处 理 流程 ， 让 读者 对 
CodeIgniter 有 一 个 更 加 直观 的 认识 。 


4.4.2 安装 Codelgniter 


CodeIgniter 是 一 套 由 PHP 5 所 构建 的 PHP MVC 框架 ， 要 使 用 CodeIsgniter 并 不 需要 特殊 
的 环境 配置 ， 也 不 需要 安装 额外 的 PHP 扩展 (Zend Framework, CakePHP 等 主流 框架 均 需 
要 安装 PDO )， 只 需要 具备 PHP 5.x 运行 环境 即 可 。 读 者 可 以 在 http://codeigniter.org.cn/ 
downloads 2tTD. CodeIgniter 安装 包 ， 解 压 后 得 到 的 文件 如 图 4-20 所 示 。 
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hh o 


.Settings applicatio system user guid index.php 


a Go 


e 


license.txt .project .buildpath 


图 4-20 CodeIgniter 框架 文件 


如 图 中 所 示 ，system HENX CodeIgniter 框架 的 核心 代码 包 ，application 为 网 站 应 用 日 
录 ， 是 开发 人 员 主 要 处 理 的 目录 ，application 目录 结构 如 图 4-21 所 示 。 


mme 


E 


, cache 缓存 
config 应 用 配置 文件 
,Controllers 一 一 一 一 一 > MVC 控 制 器 
core 应 用 核心 类 库 
s errors 一 一 一 一 一 一 一 > 异常 容器 
5 helpers 一 一 一 一 > 应 用 核心 类 库 
, hooks 控制 器 扩充 容器 
s language 一 一 一 一 一 一 > 多 语言 
, libraries 一 一 一 一 一 一 > 官方 及 第 三 方 类 库 
, logs 网 站 应 用 日 志 
, models MVC 模 型 
,third_party 目 定义 扩展 模块 
s views MVC 视 图 
index.html 


.htaccess | 


图 4-21 application 目录 结构 


将 解压 后 的 文件 复制 到 网 站 根 目 录 ， 例 如 Codelgniter 目录 ， 此 时 通过 http://localhost/ 
CodeIsniter 网 址 ， 束 可 以 初始 化 开发 环境 。CodeIsniter 是 单 入 口 模式 、 单 应 用 的 开发 框架 ， 
读者 可 以 将 域名 绑 定 到 Codelgniter 文件 夹 。 通 过 前 面 的 步骤 ，CodeIgniter W ZRT T, 
接 下 的 MVC 开发 步骤 均 在 application 目录 中 完成 。 


4.4.3 使 用 Codelgniter 实现 MVC 


接 下 来 将 使 用 Codelgniter 实现 一 个 简单 的 狐 闻 列表 ， 用 户 点 击 列表 中 的 新 闻 标 题 将 会 
进入 新 闻 的 正文 页 面 ， 通 过 该 例子 演示 Codelgniter 实现 MVC 开发 的 流程 。 


1. 创建 数据 表 


为 了 方便 演示 CodeIgniter 模型 的 调用 机 制 ， 新 闻 标 题 将 使 用 数据 表 存 放 。 首 先 在 
MySQL 中 创建 名 为 codeigniter 数据 库 ， 然 后 创建 一 个 news 表 ，SQL 代码 如 下 。 


CREATE TABLE 'codeigniter'.'news' ( 
'id' TINYINT NOT NULL AUTO_INCREMENT PRIMARY KEY , 


'title' VARCHAR( 60 


) NOT NULL , 


"Content ” TEXT NOT NULL , 
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'add_time' INT( 9 ) NOT NULL , 


INDEX ('title') 


) 
接着 问 news 表 添 加 新 闻 数 据 。 


INSERT INTO 'codeigniter'.'news' ( 
We 
"e relet, 
Teontenti 
'add time' 
) 
VALURS 
NULL ， "BRIE, BRIANA, '1346401694' 
Jy A( 
NULL ， "第 二 条 新 闻 标 题 ' ， "第 二 条 新闻 内 容 '，'1346401704:! 


) 8 


创建 完 数据 库 和 表 之 后 ， 还 需要 更 改 Codelgniter 的 数据 库 连 接 配置 参数 ， 打 开 
， 修 改 该 文件 的 配置 参数 ， 代 但 如 下 。 


application/config/database.php 文件 


Şactive_group = 'default'; 
Sactive record = TRUE; 


şdb['default'] ['hostname'] = 
sdb['default']['username'] = 
şdb['default'] ['password'] = 
sdb['default']['database'] = 
Se GE 
Eeer ele 
şdb['default'] ['pconnect'] = 
sdb['default']['db debug'] = 
selol esSEaulE ] I cache omt] = 
Selo l “aertawle a Ee Ee Reg 
EE ee 
Selol "oeraule"| Lee eh 
şdb['default'] ['swap_pre'] = 
Seloll Veereawlke” j ee Ee bes 
Stelle e ai ] [V str ricrtron" ] 


E Ee 
I Ot 
OO 
codeigniter" p 
“mysa” 8 
LD Wi 

ri 
TRUE; 
TRUE; 
FALSE; 
lB 

lÁ 
Vacon; 
merere General Ci“ p 
EE 

lÁ 
TRUE; 
FALSE; 


以 上 配置 信息 读者 需要 根据 MySQL 实际 环境 而 定 ， 其 中 较为 重要 的 是 
$db['default']['dbdriver'] 和 $db['default']['char_set'] 。 闻 者 指定 连接 驱动 ，Codelgniter 文 持 
Oracle, MySQL, PostgreSQL 等 多 种 驱动 ;， 后 者 指定 数据 库 的 查询 编码 ， 对 于 MySQL 必须 
要 指定 查询 编 公 。 

2. 创建 模型 

接 下 将 创建 news 数据 表 对 应 的 实体 模型 ， 进 入 application/models 目录 ， 创 建 一 个 
MNews.php 模型 文件 ， 该 文件 内 容 如 以 下 代码 所 示 。 


<?php 

class MNews extends CI Modell 
// 使 用 构造 函数 
function _ construct () 


{ 


Parent:: construct (); 
i 
// 获 取 痢 闻 标 题 数 据 


function get_news_all(){ 
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$sql="SELECT * FROM news"; 
Squery= $this->db->query ($sql); 
return $query->result (); 


} 


R 


如 以 上 代码 所 示 ， 模 型 的 名 称 必 须 首 字母 大 与 ， 并 且 继 承 于 CI Model ZS, ` constructo) 
是 模型 的 构造 函数 ， 该 函数 将 实例 化 基 类 CT Model 的 构造 函数 ， 这 样 才能 在 当前 模型 中 使 
用 构造 函数 (否则 不 能 使 用 构造 函数 )。get_news_all0 用 于 获取 news 表 中 的 所 有 新 闻 标 题 ， 
其 中 $this->db->query0 指 定 使 用 CI_Model 提供 的 数据 库 查 询 方法 。 

3. 创建 控制 器 

创建 模型 后 ， 还 需要 创建 控制 器 ， 然 后 调用 模型 中 的 方法 ， 最 终 将 结果 显示 到 视 网 界 
HE. 3A application/controllers 目录 ， 创 建 一 个 名 为 news.php 的 控制 笑 文 件 ， 代 人 码 如 下 
所 示 。 


<?php 
class News extends CI_Controller{ 


public function index (){ 
$this->load->model ('MNews',"",true); 
Ş$data["list"]=$this->MNews->get_news_all(); 


// 定 义 页 面 标题 
sdata[l"page titlen]=" 新 闻 中 心 首 页 "; 
ye 


Şthis->load->view ("head", $data); 
$sthis->load->view ("index", $data); 
// 载 入 页 面 底部 
EE EE 

} 


public function content (){ 


} 
} 


如 以 上 代码 所 示 ， 一 个 合法 的 控制 费 类 名 必须 为 首 字 母 大 写 ， 类 名 与 控制 占 文 件 名 同 
名 ， 并 且 必 须 继承 于 CI Controller 其 类 。 在 Codelgniter 中 ， 控 制 器 命名 规则 比较 灵活 ， 它 
不 需要 加 前 级 或 后 经， 一 个 普通 的 文件 即 可 。$this->load->model0 用 于 载 入 模型 ， 该 方法 共 
有 3 个 参数 ， 第 1 个 参数 指定 模型 的 名 称 ， 第 2 个 参数 指定 模型 的 配置 信息 ， 即 数据 库 配 置 
言 妨 ， 如 果 留 空 将 使 用 application/config/database.php 全 局 文件 配置 信息 ; 第 3 个 参数 指定 数 
据 库 连接 的 初始 化 状态 ， 默 认 情 况 下 Codelgniter 并 不 会 上 自动 连接 数据 库 ， 需 要 将 该 参数 设 为 
true 模型 才 会 连接 指定 的 数据 库 。 

d. 创建 视图 

在 MVC 设计 中 ， 一 个 动作 就 代表 一 个 页 面 。 以 前 面 创建 的 index 动作 为 例 ， 它 对 应 的 
视图 文件 就 是 index.php。Codelgniter 默认 的 视图 文件 以 .php 作为 视图 文件 ， 开 发 人 员 可 以 在 
视图 中 使 用 标准 的 PHP 人 代码。 当然 Codelgniter 内 置 了 parser 视图 引擎 ， 支 持 使 用 .html 文件 
作为 视图 模板 。 此 外 ，CodeIgniter 还 能 以 扩展 的 方式 文 持 Smarty 作为 模板 引擎 ， 但 出 于 性 
能 考虑 ，Codelgniter 只 建议 开发 者 使 用 标准 的 PHP 文件 作为 视图 文件 。 


O O 93 


PHP MVC 开发 实战 


进入 application/views 目录 ， 在 该 目录 中 创建 一 个 名 index.php 的 视图 文件 ， 代 码 如 下 
Dirr, 

<?php foreach($list as Ş$item):?> 

<li><a href="news/content/<?php echo S$item->id;?>"><?php echo S$item->title;?> 
</a></li> 

<?php endforeach; ?> 

index 动作 还 使 用 了 head 及 foot 视图 文件 ， 这 两 个 视图 用 于 实现 网 站 整体 布局 。 分 段 视 
图 文件 载 入 不 仅 利 用 网 站 布局 ， 还 能 实现 页 面 的 局 缓存 ， 提 避 页 面 的 载 入 速度 。 
application/views/head.php 代码 如 下 所 示 。 

<html> 

<head> 


<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
<title><?php echo Ş$page_title;?></title></head> 
<body> 


application/views/foot.php 代码 如 下 所 示 。 
</body> 
</html> 


D Eie ARA Codelgniter 视图 处 理 过 程 ， 在 实际 开发 中 可 根据 需求 在 head.php 中 
定义 页 面 的 布局 及 样式 代码 。 最 后 打开 http://localhost/Codelgniter/index.php/News 网 址 ， 将 
能 够 查看 到 news 数据 表 中 的 新 闻 标 题 数 据 。 


4.5 ”高 效 便 捷 的 ThinkPHP 


4.5.1 ThinkPHP 介绍 


ThinkPHP 是 一 套 国内 著名 的 PHP MVC FRIE, CREER RIEA Web 应 用 的 开发 
周期 ， 典 型 的 单一 入 口 文件 模式 能 够 提供 友好 的 URL， 使 得 网 站 更 容易 地 被 搜索 引擎 收录 。 
ThinkPHP 经 过 了 六 年 的 发 展 ， 当 前 最 新 版 本 为 3.0。 最 新 版 本 提供 了 NoSQL、 云 技术 、 分 
布 式 文 持 ， 使 得 ThinkPHP 的 功能 更 加 完善 ， 能 够 满足 大 型 Web 应 用 的 开发 需求 。 

ThinkPHP 借鉴 了 著名 的 Struts (Java Web 应 用 开发 框架 ) 思想 ， 实 现 了 MVC 严谨 的 分 
层 设计 思路 、 对 象 关 系 映 射 CORM)、 数 据 库 CURD HESE., ThinkPHP 提供 了 比 Smarty 还 
灵活 和 好 用 的 视图 引擎 ， 能 够 将 页 面 设计 师 和 PHP 程序 员 分 离 ， 保 证 网 站 的 质量 。 

由 于 ThinkPHP 的 灵活 和 高 效 ， 经 过 这 几 年 的 发 展 ，ThinkPHP 已 经 成 为 国内 最 受 欢迎 的 
PHP MVC 框架 。 如 果 是 个 人 或 何 单 的 项 目 ， 使 用 ThinkPHP 并 不 会 造成 效率 降低 的 问题 ; 
作为 公司 或 者 团队 项 目 ，ThinkPHP 不 仅 能 够 提高 开发 效率 ， 内 置 的 数据 库 集 群 功能 、 多 应 
用 分 离 功能 ， 还 能 支持 超大 规模 的 网 站 开发 。ThinkPHP 内 置 的 多 种 缓存 模式 、 代 码 预 编译 
机 制 ， 方 便 多 用 的 模块 静态 化 使 得 网 站 的 运行 速度 得 到 质 的 提高 ， 这 对 于 大 型 网 站 开发 而 言 
是 尤其 重要 的 。 

ThinkPHP 虽然 由 国人 开发 ， 但 并 不 意味 大 ThinkPHP 不 适合 国际 化 开发 ， 事 实 上 
ThinkPHP 从 2.0 开始 就 提供 了 完善 的 国际 化 文 持 ， 与 Symfony, CakePHP 等 主流 框架 一 样 
提供 多 语言 、 多 终 问 文 持 。ThinkPHP 定位 于 轻 量 级 ， 快 速 和 简单 是 该 框 染 的 主要 特点 ， 这 
在 本 书后 面 的 章节 内 容 中 将 会 明显 地 感受 到 。 接 下 来 首先 对 ThinkPHP 人 处理 MVC 的 流程 作 
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一 个 简单 的 介绍 ， 加 深 对 ThinkPHP 的 直观 认识 。 


4.5.2 TRZA ThinkPHP 


和 其 他 PHP MV 框架 一 样 ， 要 使 用 ThinkPHP 20 EI WR AE Ee 0 CT, wA 
ThinkPHP Fil% Codelgniter 一 样 徐 单 ， 只 需要 PHP 5.x 环境 即 可 。ThinkPHP 文 持 多 种 数据 
库 驱 动 ， 包 括 PDO 套件 ， 接 下 来 将 通过 ThinkPHP 实现 一 个 简单 的 MVC 应 用 。 

1. 下 载 ThinkPHP 

读者 可 以 在 http://thinkphp.cn/down.html 页 面 上 下 载 到 ThinkPHP 压缩 包 文 件 ， 也 可 以 在 
http://thinkphp.googlecode.com/svn/trunk SVN 代码 库 中 获取 到 ThinkPHP 最 新 源 代 人 码 文 件 。 这 
里 将 下 载 的 版 本 为 ThinkPHP 3.0， 解 压 后 ThinkPHP 目录 结构 如 图 4-22 所 示 。 


L Common ThinkPHP j E, 
l. Conf ThinkPHP 配 置 文件 
|. Extend 一 一 一 一 一 ThnkPHP 扩 展 包 
[L Lang 一 一 一 一 一 一 一 ThinkPHP 语 言 包 
L Lib ThinkPHP H LHE 
用 Tpl ThinkPHP tR 
| README.txt 
W logo.png 
| LICENSE.txt 


回 ThinkPHP.php 


ThinkPHP 人 入 口 文件 
图 4-22 ThinkPHP 目录 结构 


如 图 中 所 示 ，ThinkPHP 3.0 的 目录 结构 很 少 ， 关 键 的 核心 包 文件 位 于 Lib HKF, SH 
KER ST MVC 控制 器 、 数 据 库 驱动 、 模 板 引 擎 、 网 络 通信 类 、 文 件 处 理 等 ， 对 于 MVC H 
发 人 员 而 言 ， 并 不 需要 太 多 地 关注 ThinkPHP 本 身 的 这 些 类 库 ， 因 为 ThinkPHP 提供 了 完善 
的 自 定 义 扩 展 ，ThinkPHP 核心 类 库 将 在 5.2.1 章节 中 介绍 。 

2. 安装 ThinkPHP 

要 使 用 ThinkPHP 非常 简单 ，ThinkPHP 并 不 需要 安装 ， 所 谓 的 安 帮 只 不 过 是 引入 
ThinkPHP 入 口 文件 即 可 。 假 设 网 站 目录 为 tp, MWA RAHM ThinkPHP 文件 夹 复 制 到 tp H 
录 下 ， 然 后 在 tp 目录 下 创建 项 目 入 口 网 址 ， 此 处 所 创建 的 入 口 文件 为 index.php， 代 人 码 如 下 
所 示 。 


<?php 

define (STAHINK PATEY hl ey 
define (TAPE PATH", H. /ZBome/ni: 

Cerine E EE WinCE) y 

// 引 入 ThinkPHP 入 口 文件 ,初始 化 应 用 状态 
require_once (THINK_PATH."ThinkPHP.php"); 
?> 


如 上 述 代 码 所 示 ， 共 定义 了 3 个 系统 常量 : THINK_PATH 用 于 指定 ThinkPHP 上 所 在 目 
录 ; APP_PATH 指定 了 当前 项 目 存放 目录 ; APP NAME 指定 了 当前 项 目 名 称 ， 用 于 多 项 目 
部 署 〈 如 前 台 和 后 台 )。 部 署 完成 后 ， 访 问 http:/localhost/tp 网 址 即 可 初始 化 开发 环境 ， 
ThinkPHP 会 目 动 创建 必要 的 目录 结构 和 文件 ， 如 图 4-23 所 示 。 
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Firefox ~ 


图 4-23 ThinkPHP 成 功 初 始 化 项 目 


4.5.3 ”使 用 ThinkPHP 实现 MVC 


为 了 便于 学 习 和 测试 ， 读 者 可 以 根据 4.3.3 节 内 容 将 tp 目录 绑 定 为 一 个 虚拟 网 站 ， 这 里 
绑 定 的 域名 为 http:/tp.localhost， 本 书后 面 的 章节 内 容 如 无 另外 说 明 均 在 该 虚拟 网 站 中 完成 。 
geen ThinkPHP 创建 一 个 简单 的 MVC 网 站 应 用 。 
. 创建 数据 库 
SE ThinkPHP 完整 的 MVC 处 理 流程 ， 首 先 创 建 一 个 数据 库 和 一 个 数据 表 ， 
分 别 命名 为 tp 数据库) 和 tpk_user( 数 据 表 )，tpk_user 数据 表 SQL 代码 如 下 。 


CREATE TABLE Ui OR 
'id' TINYINT NOT NULL AUTO INCREMENT PRIMARY KEY , 
'user_name' CHAR( 20 ) NOT NULL , 
'user email' CHAR( 40 ) NOT NULL , 
INDEX ('user_name' , 'user_email' ) 
) ENGINE = MYISAM ; 


然后 问 t gea user KPIA H SS SO, SQL 代码 如 下 所 示 。 
ENEE EE ( 
Ve DE 


'user_name' , 


'user_ email' 
) 
VALUES ( 
Ne 0 


jo Ek 
NUE ea VCeiba Q26 Con 


) 7 

2. 修改 配置 文件 

ThinkPHP 默认 的 数据 库 驱 动 即 为 MySQL， 所 以 无 须 做 任何 修改 即 可 连接 到 MySQL 数 
据 库 。 打 开 项 目的 配置 文件 home/Conf/config.php， 增 加 MySQL 数据 库 连 接 信 息 ， 如 以 下 代 
Dir a. 


<?php 


return array 人 
// "配置 项 "=>" 配 置 值 ' 
'URL_CASE_INSENSITIVE' =>true, 


(pp TP => 'mysql', // 数据 库 类 型 
DB HOST. E // DRSSenht 
'DB_NAME' => 'tp', // 数据 库 名 
IDEU ERS => 'root', // 用 户 名 
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'DB_PWD' => 'root', // ZW 
'DB_PREFIX' => 'tpk_', // 数据 库 表 前 绥 
) 8 


?> 


通过 正确 修改 上 述 配置 文件 信息 ， 现 在 ThinkPHP 就 可 以 操作 MySQL 数据 库 了 。 需 要 
注意 的 是 ， 由 于 ThinkPHP 默认 会 使 用 预 编译 机 制 ( 即 在 home/Runtime 目录 里 生成 
~runtime.php 文件 )， 所 以 每 次 修改 完 配 置 文件 后 均 需 要 删除 ~runtime.php 文件 (可 以 直接 清 
"7 home/Runtime 目录 )， 新 的 配置 数据 才 会 生效 。 

3. 创建 模型 

根据 MVC 的 设计 思想 ， 模 型 (Model) 是 用 于 影射 数据 表 的 ， 所 有 CURD 应 该 都 在 
Model 中 完成 。 但 ThinkPHP 有 些 不 同 ，ThinkPHP 允许 开发 人 员 直 接 在 控制 器 中 完成 这 些 操 
作 ， 不 必 拘 泥 于 在 日 定义 的 Model 中 完成 (当然 也 支持 在 自 定 义 Model 中 完成 )。 接 下 来 将 
创建 一 个 User 模型 ， 并 在 该 模型 中 实现 数据 但 询 操 作 。 


<?php 


class UserModel extends Modelt 
function get user(){ 
ŞuserOobj=M ("User"); 
Şrows=$user0Obj->select(); 


return $rows; 


| 


在 ThinkPHP 中 无 论 是 模型 的 文件 名 ， 还 是 模型 的 类 名 都 必须 遵循 内 建 的 文件 规则 。 如 
UserModel.class.php XF, User 代表 模型 名 称 ，Model 为 模型 文件 的 标识 ，.class.php 为 
ThinkPHP 框架 所 规范 的 文件 后 级 名 ; UserModel 为 类 名 称 ， 所 有 用 户 自 定义 模型 都 必须 继承 
于 Model 或 者 AdvModel 基础 模型 类 。 

4. 创建 控制 器 

在 ThinkPHP 中 ， 控 制 器 的 命名 规则 和 模型 相同 ， 控 制 器 的 文件 标识 为 Action。 接 下 来 
将 在 home/Lib/Action 目录 中 创建 一 个 Member 控制 器 ， 文 件 名 为 MemberAction.class.php， 
代码 如 下 。 


<?php 


class MemberAction extends Action{ 

public function index (){ 
// 实 例 化 自 定 义 模型 
SuserObj=D ("User"); 
// 调 用 oer user 方法 
Şrows=$user0Obj->get_user();}; 
// 将 结果 分 配 级 视图 变量 data 
sthis->assign ("data", $rows); 
// 显 示 视 图 模板 
Şthis->display ();}; 


} 


如 上 述 代码 所 示 ， 在 index 动作 中 调用 了 前 面 创建 的 User 自 定义 模型 ， 该 模型 中 有 一 个 
get_user(0) 方 法 ， 该 方法 将 获取 tpk_user 表 中 的 用 户 数据 。$this->assign("data",$rows) 指 定 了 将 
数据 输出 到 视图 模板 ， 接 下 来 需要 创建 对 应 的 视图 模板 。 


D Rn 


PHP MVC 开发 实战 


5. 创建 视图 模板 

进入 home/Tpl 目录 ， 在 该 目录 中 创建 一 个 分 类 Member， 一 个 控制 器 对 应 一 个 模板 分 
类 ， 然 后 进入 Member 目录 ， 在 该 目录 中 为 index 动作 创建 视图 模板 ， 并 命名 index.html 
(ThinkPHP 项 目 模板 默认 使 用 .html 作为 视图 模板 )， 代 码 如 下 所 示 。 


<table width="780" height="107" border="0" cellspacing="1" bgcolor=" #CCCCCC"> 
KEES 
<td heignt= "42" align="center" bgcolor "#FFEFFE" ES dE 
wd alionscfoeentert PIColor VYEEFERRY Hl DS td 
<td align="center" bgcolor="#FFFFFF"> 用 户 邮 箱 </td> 
</tr> 
<volist name="data" id="vo"> 
LTES 
<td height="62" align="center" bgcolor=" #FFFFFF">{$vo.id}</td> 
<td align="center" bgcolor="#FFFFFF">{$vo.user_name}</td> 
<td align="center" bgcolor=" #FFFFFF">{$vo.user_email}</td> 
SE 
</volist> 
</table> 


如 以 上 代码 所 示 ， 由 于 在 index 动作 中 分 配给 视图 模板 的 数据 为 数组 ， 所 以 需要 使 用 循 
环 语 句 裔 历 变 量 数 据 。 在 ThinkPHP3.0 中 ， 可 以 使 用 <volist> 或 <foreach> 标 签 表 历数 组 变 
量 ， 使 用 方式 类 似 于 Smarty 中 的 foreach 标签 。 同 时 也 支持 直接 使 用 标准 的 PHP TU. 

由 于 volist 提供 了 非常 灵活 的 特性 ， 并 且 很 好 地 将 界面 与 多 辑 进 行 分 离 ， 所 以 应 该 尽量 
避免 在 视图 代码 中 嵌入 PHP 代码 。 关 于 ThinkPHP 的 视图 设计 ， 在 第 6 章 6.4 节 中 会 详细 介 
绍 ， 在 此 读者 只 需要 了 解 其 作用 即 可 。 最 终 运行 结果 如 图 4-24 所 示 。 


seh 
-Insert title here x | 国 XAMPP 1.7.4 x Las localhost / localhost / tp / ... * 
< > H H G /@tp.localhost/index.php; Yr Y C |W- Aam pg D- ei, dë, Ges 


用 户 邮箱 


kf@86055.com 


ceiba_@126.com 


图 4-24 ThinkPHP MVC 运行 效果 


D 说 明 : 在 ThinkPHP 3.0 以 前 的 版 本 中 ， 视 图 还 有 主题 的 概念 ， 所 谓 的 主题 是 为 了 使 网 站 更 好 地 应 用 多 
模板 、 多 风格 。 比 如 一 个 网 站 也 许 会 有 红色 风格 和 黑色 风格 ， 但 必须 要 有 一 个 默认 风格 ， 所 以 
ThinkPHP 2.x 在 创建 应 用 时 会 在 Tpl 目录 中 创建 一 个 defaut 目录 ， 该 目录 即 为 默认 的 模板 风格 。 以 本 
"DS ul. Member 模板 分 类 应 该 处 于 defaut 风格 下 ， 所 以 index.htm 的 路 径 应 该 为 home/ 
Tpl/default/Member/index.html。 在 ThinkPHP 3.0 中 并 不 是 没有 主题 的 概念 ， 只 是 默认 的 主题 为 空 ， 如 果 
确实 需要 多 主题 ，DEFAULT_THEME 配置 参数 可 以 指定 当前 应 用 的 主题 目录 。 


4.5.4 ”高 效 的 ThinkPHP 视图 引擎 
前 面 已 经 简单 介绍 过 ThinkPHP 视图 引擎 ， 这 些 标签 处 理 机 制 类 似 于 JSP tag, EH 
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XML 作为 标签 的 泻 染 方式 ， 开 发 人 员 可 以 方便 地 在 网 员 中 矢 入 XML 标签 ， 然 后 由 
ThinkPHP 视图 引擎 解释 成 标准 的 PHP 代码 。ThinkPHP 的 视图 引擎 之 所 以 高 效 ， 主 要 体现 
在 它 的 标签 扩展 性 。ThinkPHP 的 视图 标签 扩展 性 够 完美 地 与 后 合 控制 右 代 但 相 结合 ， 例 如 
在 Smarty 中 使 用 函数 功能 ， 首 先 需 要 在 后 台 PHP 代码 注册 ， 而 在 ThinkPHP 中 完全 不 需 
要 ， 如 以 下 代码 所 示 。 


< ant st ne o> 


<ed name="vo['ClassPid']" value="$id"> 
<div class="post"> 
<p class="meta"><span class="date"><a 
href="{$Think.config.SYS URL}/c/{$vo[l"id"]}.html">{$vo[l"ClassName"] }</a></span> </p> 

<div class="entry"> 

{:resColumn ($vo['id'],6)} 
</div> 

</div> 


</eq> 
</volist> 


细心 的 读者 也 许 已 经 发 现 ， 在 前 面 的 内 容 中 并 没有 像 CakePHP、Symfony 那样 的 网 
站 布局 概念 ， 事 实 上 ThinkPHP 已 经 提供 了 多 种 方式 解决 网 站 布局 的 问题 ， 其 中 第 用 的 有 
直接 在 模板 中 使 用 <include file="" /> 标签 ， 或 者 在 控制 器 中 直接 分 段 输出 模板 (类 似 于 
CodeIgniter 方式 )。 相 对 而 言 使 用 <include file="" /> 标签 更 加 友好 ， 也 更 加 符合 ThinkPHP 
的 模板 处 理 机 制 特点 ， 以 下 代码 将 使 用 <include file="" /> 标签 包含 网 站 的 head 和 foot, 
代码 如 下 所 示 。 


<html> 
<head> 


<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 

L<eicle>igrtciclel<s/ticle> 

<meta name="description" content={$Think.config.SYS Name}" /> 

<script src="{$Think.config.SYS_URL}/Publice/js/jquery.js"></script> 

<link href="{$Think.config.SYS_URL}/Publice/style.css" rel="stylesheet" type="text/ 
css" media="screen" /> 

</head> 

<body> 

<include file="./Publice/head.html" /> 

<div id="page"> 

</div> 

<include file="./Publice/foot.html" /> 

</body> 

</html> 


上 述 代码 有 一 定 的 局 限 性 ， 但 能 够 提供 非 音 清晰 的 思路 ， 对 界面 设计 人 员 非 常 友 
W, include 标签 不 仅 可 以 直接 包含 网 页 ， 还 可 以 包含 当前 项 目 探 制 器 中 的 方法 ， 例 如 
<include file="Public:head"/> 即 表示 包含 日 定义 控制 占 Public 中 的 head 334E (CD, 
然 使 用 分 段 输出 模板 也 很 方便 ， 这 种 方式 比较 适合 PHP 和 界面 混合 编程 的 团队 ， 如 以 下 
代码 所 示 。 


Şthis->display ('Public:head'); 
Epis EE EE 
HE SE (Publics roor) p 
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4.6 ”小 结 


本 章 着 重 介 绍 了 当前 国内 外 比较 主流 的 几 款 PHP MVC HEX, XE MVC 框架 是 PHP 
MVC 中 最 经 典 和 最 完善 的 编程 框架 。 尤 其 是 Zend Framework 和 Symfony， 是 PHP 程序 员 学 
习 MVC 开发 的 风 癌 标 ， 它 们 完善 的 架构 和 先进 的 设计 思想 能 够 提高 PHP 应 用 质量 。 本 章 最 
后 还 介绍 了 国内 人 气 最 高 的 ThinkPHP， 该 框架 无 论 是 框架 本 身 功 能 、 代 码 编写 习惯 还 是 帮 
助 文 档 支 持 都 非常 适合 国内 开发 人 员 使 用 ， 也 是 一 球 值 得 所 有 热爱 开源 程序 需要 关注 的 PHP 
MVC 框架 。 

当然 ， 本 章 介 绍 的 PHP MVC 框架 并 不 能 概括 所 有 PHP MVC 框架 的 特性 ， 但 万 变 不 
离 其 宗 ， 理 解 透 了 其 中 一 款 MVC 的 设计 思路 ， 再 面 对 其 他 PHP MVC 框架 就 游 才 有 余 
了 。 基 于 国内 环境 考虑 ， 本 书后 面 内 容 将 着 重 对 ThinkPHP 3.0 进行 深入 的 讲解 ， 这 些 内 容 
由 浅 入 深 、 简 单 易 懂 ， 能 够 迅速 帮助 PHP 程序 员 ， 尤 其 是 PHP 新 手轻 松 地 进入 PHP MVC 
开发 领域 。 
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内 容 提 要 


ThinkPHP 是 一 套 国内 非常 出 名 的 PHP MVC 开发 框架 ， 它 易 懂 易学 、 效 率 高 。 本 章 按 
照 循序 渐进 ， 由 浅 入 深 的 讲解 思路 ， 结 合 示例 代码 帮助 读者 迅速 上 手 。 

通过 本 章 内 容 ， 读 者 将 深入 理解 PHP MVC 编程 模式 ， 结 合 ThinkPHP 帮助 手册 ， 完 全 
能 够 开发 出 任意 类 型 的 PHP 网 站 。 需 要 说 明 的 是 ，ThinkPHP 经 过 多 年 的 发 展 ， 当 前 已 经 发 
展 到 了 3.0 版 本 。 事 实 上 ，ThinkPHP 里 程 碑 版 本 为 2.0， 该 版 本 竟 定 了 ThinkPHP 的 核心 架 
构 ， 提 供 了 适合 企业 级 开发 的 众多 特性 ， 所 以 现在 众多 团队 还 在 使 用 ThinkPHP 2.x 版 本 构建 
网 站 。 

新 版 本 与 2x 版 本 相 比 主要 是 在 应 用 的 部 闭 方 式 上 做 了 一 些 修 改 ， 并 且 全 面 使 用 CBD 
模式 (核心 Core+ 行 为 Behavior+ 驱 动 Driver )， 增 强 了 安全 性 ， 但 在 应 用 开发 层面 变化 并 不 
大 ， 所 以 接 下 来 的 内 容 同 样 也 适合 ThinkPHP 2.x FRAR. 

本 章 将 会 围绕 ThinkPHP 核心 功能 包 进 行 讲解 ， 包 括 框架 的 文件 及 目录 组 成 ， 项 目 部 署 
方式 等 。 通 过 这 些 内 容 的 介绍 ， 为 后 续 扩 展开 发 打下 基础 。 

学 习 目标 
了 解 ThinkPHP 构建 MVC 应 用 的 方式 。 

了 解 ThinkPHP 系统 目录 及 应 用 目录 结构 。 


掌握 ThinkPHP 创建 项 目的 全 过 程 。 
了 解 MVC 配置 文件 作用 及 创建 方式 。 
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5.1 大 道 至 人 简 、 开 发 由 我 


通过 前 面 第 4 章 4.5 节 内 容 的 介绍 ， 相 信 读 者 已 经 对 ThinkPHP 有 了 和 初步 的 认识 。 
ThinkPHP 无 论 是 从 应 用 部 署 方面 ， 还 是 代码 组 织 方面 都 显得 非常 灵活 。 常 见 的 PHP MVC 
框 染 通常 都 将 PHP WEA OJAD 放置 到 特定 的 目录 中 (例如 CakePHP 放 到 app/webroot H 
录 ; CodeIsniter 放 到 application 目录 )， 而 ThinkPHP 不 需要 指定 存放 目录 ， 它 可 以 放置 到 任 
意 的 网 站 目录 中 ， 需 要 的 只 是 一 个 入 口 文 件 。 在 入 口 文 件 中 完成 框架 所 有 初始 化 操作 。 需 要 
说 明 的 是 ， 入 口 文 件 通 常 命名 为 index.php， 但 并 非 一 定 是 index.php 文件 ， 也 可 以 目 定义 入 
口 文件 名 。 入 口 文件 是 ThinkPHP 应 用 的 入 口 ， 是 一 个 项 目 中 最 重要 的 文件 ， 所 以 是 必需 
的 ， 下 面 首 先 介绍 ThinkPHP 项 目 入 口 文件 。 


5.1.1 入 口 文 件 


入 口 文件 用 于 初始 化 项 目 。 应 用 从 运行 开始 到 运行 结束 ， 整 个 生命 周期 都 由 入 口 文 件 管 
理 。 如 果 人 入口 文 件 配置 出 现 错误 ， 那 么 整个 应 用 将 只 俘 留 在 入 口 文件 ， 无 论 用 户 怎样 请 求 后 
面 的 结 朱 都 将 失效 。 这 征 单一 入 口 框 民 的 共同 特性 ，ThinkPHP 入口 文件 文 持 两 种 项 目 部 普 
方式 : 一 种 是 传统 的 一 个 入 口 文件 对 应 一 个 应 用 的 方式 ; 男 一 种 是 所 有 项 目 对 应 一 个 入 口 文 
件 〈 即 应 用 分 组 模式 )。 这 里 首先 介绍 第 一 种 入 口 文件 模式 ， 代 码 如 下 。 


<?php 
define (THINK PATH, ™./TNIinkPHE/ Y); 
define (TABE BATH", H. Zhome/ni: 


define ("APP_NAME", "index"); 
define ("APP DEBUG", true); 


require_once (THINK_PATH."ThinkPHP.php"); 


上 述 代码 中 ， 使 用 了 3 个 预定 义 常 量 定义 项 目的 初始 化 参数 。 其 中 APP_PATH 定义 项 
H H; APP_NAME 定义 项 目 名 称 ; APP_DEBUG 定义 调试 模式 。 这 3 个 常量 都 是 可 选 
的 ， 最 简单 的 入 口 文件 只 需要 引入 ThinkPHPphp 文件 即 可 ，ThinkPHP 会 自动 生成 项 目 文 件 
与 目录 结构 ， 如 图 5-1 ra, 
| A) 
Lib Tpl 


Common Conf Lang Runtime 


图 5-1 home 项 目 目录 结构 


在 入 口 文件 中 ，ThinkPHP 规范 了 一 些 系 统 音量 ， 在 入 口 文件 中 定义 这 些 种 量 值 可 以 修 
改 项 目的 属性 。 例 如 前 面 创建 的 index.php 开启 了 APP_DEBUG， 此 时 整个 应 用 下 的 所 有 控 
制 器 都 将 输出 调试 信息 。 入 口 文件 第 见 的 预定 义 第 量 如 表 5-1 所 示 。 

项 目 一 旦 完成 初始 化 ， 接 下 来 的 操作 将 交 由 ThinkPHP 人 处理， 最 终 的 运行 结果 也 由 入 口 
文件 返回 ， 整 个 流程 如 图 5-2 所 示 。 
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表 5-1 入 口 文 件 常 见 预定 义 常量 


To Œ S 认 值 说 H 
URL_COMMON 1 项 目的 URL 模式 
APP_PATH null 项 目 日 录 
COMMON_PATH APP_PATH.Commony 项 目 公共 文件 目录 
APP NAME null 项 目 名 称 
APP_DEBUG flash 是 否 开启 调试 模式 
THEME_NAME null 项 目 视 图 主题 
HTML _ PATH APP_PATH.Html/ 生成 静态 文件 目录 
TEMP_PATH RUNTIME_PATH.Tempy 项 目 缓存 目录 
LANG_PATH APP_PATH.'Lang/' 语言 包 目 录 


ThinkPHP 


图 5-2 MVC 处 理 流程 


5.1.2 ”两 种 创建 项 目的 方式 


前 面 介绍 了 ThinkPHP 传统 的 项 目 部 署 方式 。ThinkPHP 还 支持 模块 分 组 方式 〈 即 所 有 项 
目 共用 一 个 入 口 文件 )， 模 块 分 组 方式 能 够 提高 项 目 之 间 的 整合 性 ， 由 于 所 有 的 项 目 都 被 放 
置 于 同一 级 目录 中 ， 所 以 彼此 之 间 的 公共 资源 〈 如 公共 函数 库 、 扩 展 类 库 、 静 态 资源 文件 
等 ) 能 够 很 好 地 被 重复 利用 。 同 时 由 于 只 有 一 个 入 口 文 件 (通常 为 index.php)， 通 以 过 配置 
URL 路 由 ， 使 得 URL 更 加 友好 。 典 型 的 多 入 口 文件 项 目 目 录 结 构 如 下 。 


ThinkPHP aas 框架 包 
Public irei 所 有 项 目 公 共 资 源 目 录 
Uploads pp 所 有 项 目 公 共 上 传 目录 


O O 103 


PHP MVC 开发 实战 


Home eee Home 项 H 

Admin .………… Admin 项 目 

index.php pp Home 项 目 入 口 文件 

admin .php eee Admin 项 目 入 口 文件 

采用 模块 分 组 后 ， 项 目的 目录 结构 将 变 得 更 加 简洁 ， 如 下 所 示 。 
E E 框架 包 

App “ee 项 目 目录 

Public eese. 所 有 项 目 公共 资源 目录 

Uploads eseese. 有 项 目 公共 上 传 目 录 

index.php +=- 所 有 项 目 入 口 文件 


采用 模块 分 组 后 ， 所 有 项 目 文件 都 被 放 到 App 目录 里 ， 这 里 的 App 目录 是 可 以 在 入 口 
文件 中 自 定 义 的 。App 目录 将 包含 Home 项 目 和 Admin 项 目 ， 结 构 如 下 。 


Common ………… Dm H A JC eg ir 


Lang eessen 项 目 公共 语言 包 


Lip ………………… 项 目 库 类 


Home en Home 项 目 控制 器 目录 
Admin pp Amin 项 目 控 制 右 目录 
RE Joe 公共 模型 目录 
Runtime pp 公共 运行 时 文件 存放 目录 
et Ee 公共 项 目 模板 目录 
Home pe Home 项 目 模板 文件 目录 
Ami pose: Admin 项 目 模板 文件 目录 
上 述 目 录 结 构 即 为 理想 状态 下 ， 一 个 向 单 的 模块 分 组 所 需要 的 目录 结构 。 如 采 使 用 多 语 
言 、 多 主题 等 ， 目 录 绪 构 的 层次 会 更 深 。 可 见 使 用 模块 分 组 方式 虽然 看 上 去 目录 结构 简单 明 
了 ， 但 实际 上 模块 分 组 的 目录 结构 都 必须 有 层 的 概念 〔 即 一 个 分 组 对 应 一 个 日 录 )。 模 块 分 
组 后 能 够 提高 整个 系统 的 密封 性 ， 也 大 大 提高 了 ThinkPHP 部 效应 用 的 容量 ， 各 项 目 之 间 可 
以 非常 容易 地 相互 调用 ， 而 不 必 担 心 系统 找 不 到 资源 和 文件 。 


5.1.3 ”模块 分 组 


前 面 简单 介绍 了 模块 分 组 的 目录 结构 ， 模 块 分 组 最 草 要 的 概念 就 是 它 的 目录 结构 发 生 了 
比较 大 的 变化 。 下 面 将 使 用 ThinkPHP 3.0 创建 一 个 简单 Web 应 用 ， 该 应 用 部 署 方式 将 使 用 
模块 分 组 。 

1. WEA DO 

使 用 模块 分 组 意味 着 一 个 项 目 只 有 一 个 入 口 文件 ， 通 常 为 index.php 文件 ， 它 与 传统 的 
多 入 口 文 件 的 配置 是 一 样 的， 预定 义 常 量 也 是 通用 的 ， 不 同 的 是 需要 配置 APP NAME 和 
APP_PATH 两 项 内 容 ， 上 只 有 这 样 才能 使 用 模块 分 组 ，ipdex.php 文件 代码 如 下 所 示 。 
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<?php 
define ('THINK_PATH', './ThinkPHP/'); 
// 定 义 项 目 名 称 和 路 径 
define ('APP_NAME', 'Myapp'); 
define ('APP_PATH', './App/'); 


define ("APP DEBUG", true); 

// 加 载 框架 入 口 文 件 

require (THINK_ PATH."ThNinkPHP .php");} 
?> 


通过 URL 访问 到 index.php 文件 ，ThinkPHP 将 会 初始 化 项 目 环境 ， 并 创建 相应 的 目录 
结构 ， 此 时 并 没有 开局 模块 分 组 功能 ， 需 要 在 公共 配置 文件 中 配置 。 

2. 配置 文件 

打开 App/Conf/config.php 配置 文件 ， 该 文件 作为 模块 分 组 后 的 公共 配置 文件 ， 和 需要 配置 
其 中 的 分 组 选项 ， 如 以 下 代码 所 示 。 


<?php 


return array 人 
'APP_GROUP_LIST'=>'Admin, Home', 
'DEFAULT_GROUP '=>'Home', 
); 


?> 


如 上 述 代 人 码 所 示 ，APP_GROUP_LIST 指定 了 分 组 列表 ， 这 里 将 项 目 分 为 Admin 和 
Home。 通 过 DEFAULT_GROUP 选项 指定 默认 分 组 〈 即 默认 载 入 的 分 组 )。 公 共 配 置 文件 修 
改 完毕 后 ， 此 时 如 果 访 问 index.php 入 口 文件 ，ThinkPHP 将 会 报错 。 这 是 因为 在 公共 配置 文 
件 中 开局 了 模块 分 组 ， 但 相应 的 分 组 目录 并 未 创建 ， 所 以 还 逢 要 手动 创建 模块 分 组 目录 。 

3. 创建 模块 分 组 目录 

首先 需要 创建 Admin 和 Home 组 的 各 日 配置 文件 ， 模 块 分 组 后 目录 是 和 项 目 名 称 相 对 
应 的 ， 所 以 需要 在 Conf 目录 下 创建 Admin 和 Home 目录 。 人 然后 分 别 在 目录 中 创建 
config.php 配置 文件 ， 这 样 Admin 项 目 和 Home 项 目 就 有 了 各 目的 配置 文件 。 

同样 App/Lib/Action 目录 也 需要 创建 Admin 和 Home 模块 分 组 目录 ， 然 后 将 默认 创建 的 
IndexAction.class.php 文件 移动 到 App/Lib/Action/Home 目录 ， 此 时 再 访问 入 口 文件 时 将 不 会 
DR, "EDD DE KO, "ui 5-3 Dir, 


Firefox ~ 
E] http://localhost/tp4/ 


é>: U DG 'eio/xeIbl- ZEP A 


^_^ Hello, 欢 迎 使 用 ThinkPHP 


图 5-3 ”模块 分 组 部 署 成 功 


通过 上 述 3 个 步 又， 就 完成 了 模块 分 组 ， 其 他 的 编程 模式 和 传统 的 部 署 方式 是 一 样 的 。 
可 见 无 论 是 使 用 传统 的 多 入 口 文件 部 署 方式 ， 还 是 使 用 单 入 口 的 模块 分 组 方式 ，ThinkPHP 
都 能 够 完美 地 解决 项 目 部 署 的 问题 。 两 种 项 目 部 署 方式 同样 简单 高 效 ， 在 实际 应 用 开发 中 读 
者 可 根据 需求 选择 其 中 一 种 。 通 常情 况 下 ， 传 统 的 模式 适合 项 目 之 间 耘 合 度 不 高 的 项 目 ， 而 
模块 分 组 方式 由 于 同一 级 目录 下 ， 各 模块 之 则 可 以 方便 调用 ， 公 共 资 源 也 能 够 彼此 利用 ( 例 
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如 模块 分 组 模式 有 公共 配置 文件 的 概念 ， 而 传统 的 模 陈 并 没有 )， 上 所 以 比较 适合 大 容量 的 项 
目 。 但 十 由 于 传统 的 部 壮 模 式 目 录 结 构 少 ， 人 简单 有 明了， 也 是 ThinkPHP 官方 推荐 的 方式 ， 所 
以 本 书后 面 的 内 容 将 使 用 传统 的 项 目 部 普 方 式 进行 讲解 。 


5.2 ThinkPHP 目录 


在 前 面 的 内 容 中 已 经 介绍 过 ThinkPHP 框架 的 目录 结构 ， 虽 然 ThinkPHP 提供 了 非常 完 
普 的 扩展 机 制 ， 但 是 这 些 扩展 的 编写 方法 都 是 基于 标准 的 OOP 设计 的 ， 所 以 如 条 不 了 解 
ThinkPHP 基础 类 库 中 的 文件 结构 ， 在 编写 ThinkPHP 扩展 时 开会 无 从 下 手 ， 接 下 来 将 看 重 
介绍 ThinkPHP 核心 类 库 文件 ， 然 后 再 简单 介绍 项 目 目 录 ， 以 便 更 好 地 理解 项 目 文件 的 组 
织 方 式 。 


5.2.1 系统 目录 


ThinkPHP 3.0 共 由 6 个 一 级 目录 所 组 成 ， 这 些 一 级 目录 下 包含 了 ThinkPHP 所 需要 的 类 
库 文 件 ， 如 图 5-4 所 示 。 


Intg 


Common Conf Extend Lang Lib 
Ta h h N 
i) 化 
Tpl README.t logo.png LICENSE.tx ThinkPHP. 
xt t php 


图 5-4 ThinkPHP 目录 


其 中 Common 为 ThinkPHP 框架 的 核心 函数 目录 ， 包 含 了 ThinkPHP 标准 函数 、 基 础 函 
数 及 运行 时 函数 ; Conf 为 框架 配置 文件 目录 ; Extend 为 框架 扩展 目录 ; Lang 为 框架 语言 
录 ; Lib 为 框架 核心 目录 ; Tp 为 框 染 视图 模板 目录 。 对 于 普通 的 开 友 者 而 言 ， 最 需要 关注 
Lib 目录 及 Extend 目录 ， 接 下 来 分 别 进 行 介绍 。 

1. Lib 目录 

Lib 目录 存放 了 ThinkPHP 最 核心 的 文件 ， 它 以 模块 化 的 方式 将 文件 组 织 在 一 起 ， 所 有 
的 文件 均 遵 循 ThinkPHP 的 文件 命名 规范 ，Lib 目录 结构 如 下 。 


Behavior oe 行为 扩展 包 
CheckRouteBehavior.class .php “oe 路 由 检测 
ContentReplaceBehavior.class .php ee 输出 模板 内 容 
LocationTemplateBehavior.class.php oe 自动 定位 模板 
ParseTemplateBehavior.class.php ee 解释 模板 
ReadHtmlCacheBehavior.class.php ee 读 取 绥 存 
ShowPageTraceBehavior.class.php eee- 在 视图 页 面 输出 Trace 调试 信息 
ShowRuntimeBehavior.class.php “ee 显示 程序 运行 时 间 
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TokenBuildBehavior.class .php oe 生成 表单 令 牌 
WriteHtmlCacheBehavior.class .php “……… 和 0 生成 文件 静态 绥 存 
Core ……………. 框架 核心 包 
Action:.class php “ee 控制 硕 基 类 
App, classe, pp ee MVC 项 目 初始 化 基 类 
Behavior.class.php “oe 行为 扩展 包 基 类 
Cache.class.php eee 缓存 管理 类 〈 包 含 文件 缓存 、 内 存 缓存 、 数 据 库 缓 存 等 待 ) 
IG php irere 数据 库 驱 动 中 间 件 
Dispatcher.class.php “venes 解释 MVC 的 URL 
Log.class.php ee Hm E HEA 
Model.class.php “ee 模型 基 类 
Think.class.php ee ThinkPHP 初始 化 其 类 
ThinkException.class.php “oe 异常 处 理 类 
View.class .php eneee 视图 管理 基 类 
Widget.class.php uenee. Widget (网 页 部 件 ) 处 理 类 
Driver “oe DN Ze E, 
Cache 
Cachebile, class, pbp enee 管理 文件 缓存 
DD …………………， 数据 库 驱 动 包 
DbMysql .class .php ee mysql_connect ch 
DbMysqli.class.php “eee Mysqli 驱动 
TagLip “ere eee 视图 引擎 标签 类 库 
TagLibCx.class.php "EE CX 标签 库 解析 类 
Template 
Template me 模板 类 库 
ThinkTemplate.class.php “eneee 模板 引擎 类 
TagLib.class.php “ee. 视图 标签 解释 类 


2. Extend 目录 

Lib 类 库 只 是 提供 了 ThinkPHP 的 基础 类 库 ， 能 够 处 理 MVC 开发 中 的 最 基础 功能 ， 通 过 
Extend 美 库 ， 可 以 实现 更 加 复杂 和 高 级 的 功能 ， 例 如 多 数据 库 驱 动 ， 多 模板 解释 引擎 、 云 计 
算 开 发 等 ， 下 面 将 对 Extend 类 库 进行 讲解 。 


Berg 控制 器 扩展 类 库 
RestAction.class.php “vene. Action 控制 器 扩展 

Behavior oe 行为 扩展 
AgentCheckBehavior.class .php “enese 网 络 代理 
BrowserCheckBehavior.class .php “ee 页 面 检查 《〈 例 如 防止 刷新 、 浏 览 器 缓存 等 ) 
CheckLangBehavior.class.php eee 检测 浏览 右 语 言 并 目 动 加 载 语言 包 
CronRunBehavior.class.php “ee 计划 任务 
FireShowPageTraceBehavior.class.php “ee 将 调试 信息 输出 到 firebug 和 firePHP 
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RobotCheckBehavior.class .php eee 检测 和 限制 搜索 引擎 和 候 虫 、 网 页 机 器 人 等 
Driver CHE 驱动 扩展 类 库 
Çaghe enpres 绥 存 驱动 扩展 
CacheApachenote.class .php “…… 和 0 Apachenote FITIK 
CacheApc.class.php eee Apc (Alternative PHP Cache) 绥 存 驱动 类 
CacheDb.class.php ……………… 数据 库 绥 存 驱动 类 
Cachebaccelerator, class, php “me eAccelerator 绥 存 驱动 类 
CacheMemcache .class .php ……………… Memcache 绥 存 驱动 类 
CacheRedis: -class php Tm Redis (NoSQL) 绥 存 驱动 类 
CacheShmop.class.php “ee shmop 缓存 驱动 类 
CacheSqlite.class .php “eee Sqlite 数据 库 绥 存 驱动 类 
CacheWincache .class .php eee WinCache ( 专 为 Windows 设计 的 缓存 加 速 器 ) 绥 存 驱 动 类 
CacheXcache.class .php “oe Xcache IK% 
Dp “eroseeeeee 数据 库 驱 动 扩展 类 库 
DbIbase.class .php ee Firebird 数据 库 驱 动 类 
DbMongo.class:php Sm Mongo (NoSQL) 数据 库 驱 动 类 
DbMssql.class .php ee MsSqlServer 数据 库 驱 动 类 
DbOracle.class.php “ee Oracle 数据 库 驱 动 类 
DbPdo.class.php “ee Pdo 驱动 类 
DbPgsql.class.php ee PostgresQL 数据 库 驱 动 类 
DbSqlite.class.php eee. Sqlite 数据 库 驱 动 类 
DbSqlsrv.class.php ee Sqlsryv 数据 库 驱 动 类 
Session “wives Session 扩展 类 库 
SessionDb.class.php ee 使 用 数据 库 方式 储存 Session DR cl 
TagLip esst. 视图 标签 扩展 类 库 
TagLibHtml.class.php “veee. Html 标签 库 驱 动 
Template “ee 模板 引擎 类 库 
TemplateEase.class.php “eee EaseTemplate 模板 引擎 驱动 类 
TemplatėéLite:class.php "ee TemplateLite 模板 引擎 驱动 类 
TemplateSmart .class .php ……………… Smart 模板 引擎 驱动 类 
TemplateSmarty.class.php “eneee Smarty 模板 引擎 驱动 类 
Engine “ee 引 获 扩展 (主要 是 第 三 方 的 开发 引擎 ， 默 认 提 供 了 新 浪 SAE 云 开发 ) 
Ee me 函数 扩展 
extend.php geheie 载 入 扩展 插件 函数 
Library oons: ThinkPHP 类 库 扩 展 
ele 基础 类 库 
he 加 密 和 解密 相关 扩展 
Base64.class.php “ee Base64 加 密 类 
Crypt .class.php ee Crypt 加 密 类 
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Des.class. Php "es Des 加 密 类 
Hmac.class .php “ee Hmac 加 密 类 
Rsa.class .php ee Rsa 加 密 类 
Xxtea.class. php ed Xxtea 加 密 类 
TO 。……………………. 文件 处 理 扩 展 
Net ……………. 网 络 处 理 扩展 
Http-class. php “ee HTTP 工具 类 
IpLocation.class.php …………… IP 管理 类 (例如 获取 、 转 换 等 ) 
UploadFile.class.php “ee 文件 上 传 类 
Util .。………………。 核心 类 库 
ArrayList.class.php zsrgkerggsrg 数组 增加 扩展 (可 以 方便 地 对 数据 生成 集合 、 排 序 、 增 删 
改 等 ) 
Authority.class.php “veee. 页 面 访 问 权 限 处 理 
CodeSwitch.class.php “eneee 代码 转换 (例如 字符 集 编码 转换 等 ) 
Cookie.class.php ee Cookie 管理 类 (增强 了 了 PHP Cookie 功能 ) 
Date, Class, php mr 日 期 时 间 管 理 类 《例如 时 间 戳 取 、 格 式 转换 等 ) 
Debug.class.php eneee 系统 调试 类 
HtmlExtractor.class .php eee 网 页 数据 提取 工具 【例如 可 以 对 页 面 中 的 特殊 标签 进 
行 过 滤 等 ) 
Image.class.php “eneee 图 片 管理 类 (例如 截图 、 生 成 水 印 等 ) 
Input .class .php “eee 输入 数据 管理 类 (可 以 对 数据 的 数据 进行 过 滤 、 转 换 等 ) 
Page.class .php “ee FHR 
RBAC.class. php “ee 基于 角色 的 数据 库 方式 验证 类 【在 实际 开发 中 并 不 常用 ) 
Session.class.php “eee Session 管理 类 (增强 了 了 PHP Session 功能 ) 
Socket .class.php “ee Socket 套 接 链接 管理 类 
Stack.class.php EEN 堆栈 的 简单 实现 
String.class.php eee 字符 串 处 理 类 (字符 截取 、 计 算 、 随 机 生成 等 ) 
Mode mre 模式 扩展 类 库 〈 指 的 是 开发 模式 ，ThinkPHP 提供 了 多 种 模式 ， 例 如 cli 命令 行 、Lite 等 ) 
Model …………………， 模型 扩展 
AdvModel.class.php enee. WARA, ETORB IMD FROTE, 200 gr GER Up 
MongoModel.class.php ee MongoDb 增 、 删 、 改 、 查 操作 类 
RelationModel .class.php “ee 关联 模型 类 扩展 (提供 更 加 高 级 的 SoL 查询 功能 ) 
ViewModel.class.php “ee 视图 模型 扩展 
(ER ThinkPHP 扩展 工具 〈 默 认 提 供 了 PhpUnit 调试 工具 、Jqueyr 等 ) 
Vendor eserse 第 三 方 扩 展 包 〈 主 要 为 视图 解释 引擎 包 ， 默 认 提 供 了 REaseTemplate、Smarty、2end 等 ) 


Lib 和 Extend 目录 包含 了 ThinkPHP 重要 的 类 库 ， 这 些 类 库 的 使 用 在 后 面 的 章节 内 容 中 

会 进行 介绍 ， 在 此 读者 只 需要 理解 即 可 。 由 于 ThinkPHP 遵循 Apache 2.0 协议 ， 意 味 着 PHP 
开发 人 员 可 以 通过 修改 ThinkPHP WARE, DIr Ont Ek, GEI 
必须 要 对 Extend 目录 下 的 扩展 类 库 要 有 深入 的 认识 。 例 如 默认 情况 下 Page.class.php 分 页 类 
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只 能 满足 于 ThinkPHP 的 3 URL 模式 ， 但 在 实际 应 用 开发 中 通常 需要 在 Ajax PASNE 
甚至 移动 终端 中 使 用 分 页 功能 ， 这 时 如 果 不 修 改 Page.class.php 文件 ， 就 难以 满足 要 求 ， 再 
比如 许多 超大 型 的 网 站 数据 库 操作 均 宕 要 读 写 分 离 、 多 类 型 数据 库 读 写 等 功能 ， 这 时 就 需要 
开发 人 员 修 改 db.class.php 文件 了 。 

AL, BIA ThinkPHP 提供 了 非常 丰 蝇 的 功能 ， 但 是 要 想 让 ThinkPHP 应 用 得 更 加 得 心 
应 手 ， 修 改 或 增加 ThinkPHP 类 库 文件 是 必须 要 做 的 一 件 事 。 


5.2.2 项 目 目 录 


只 需要 在 入 口 文件 中 定义 项 目的 存放 目录 ，ThinkPHP 在 初始 化 完成 时 会 自动 创建 相应 
的 目录 《如 采 是 Linux 系统 ， 需 要 赋 给 写 谈 权限 )。 这 些 目录 结构 是 可 以 在 入 口 文件 中 定义 
的 ， 但 无 论 是 出 于 易 用 性 还 是 安全 性 考虑 ， 一 般 情 况 下 都 不 需要 更 改 内 建 的 目录 结构 。 项 目 
目录 主要 由 公共 函数 库 、 配 置 文件 、 多 语言 、 核 心 美 库 、 运 行 时 及 模板 目录 组 成 。 在 实际 应 
用 开 友 中 通 名 还 需要 一 个 公共 资源 目录 ， 用 于 存放 css BHA js 等 文件 。 典 型 的 项 目 目录 
结构 如 下 。 
ThinkPHP aseene 框架 包 
Publice eseese 公共 静态 文件 目录 〈 该 目录 必须 确保 能 够 通过 互联 网 访问 ) 
TEEN 存放 css 样式 
js ene 存在 js 脚本 
d'et sesei 存放 视图 图 片 
Upload eese.. 存在 上 传 文件 (该 目录 需要 确定 能 够 通过 互联 网 访问 ) 
home “eee home 项 目 存 放 目 录 
Common teese home M HASERA Hx 
gommoni php pm 默认 的 函数 定义 文件 
Conf eseese home 项 目 配置 文件 目录 
config.php eee.. 常规 配置 (如 数据 库 、 项 目 属性 等 ) 


htmls .php “ee 生成 HTML 配置 


routes .php “ee 项 目 URL 路 由 配置 
Html etesse 存放 生成 的 HTML 文件 


EE EE EE Index 控制 絮语 言 包 
Z 了 一 七 W 
index.php 
BID tetten 项 目 核心 类 库 目 录 
ee 控制 器 
Ee 行为 
E soniers 模型 


Widget Sit E 部 件 
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2 公共 运行 时 模板 〔 非 调试 模式 下 项 目的 配置 信息 、 公 共 函 数 等 会 被 会 预先 


Public …………… 项 目 公 共 模 板 


dispatch_jump_success.tpl eege een 成 功 提 示 信 息 模 板 


dispatch ]jumP error .tpPp 工 pp 出 错 提供 信息 模板 
Index “rien Index {EEk (Faili) 模板 目录 
index. html “es index 动作 视图 模板 


index.php “ee home 项 目 入 文件 
项 目的 目录 结构 是 多 变 的 ， 除 了 前 面 提 到 的 项 目 部 署 方式 会 改变 目录 结构 外 ， 
ThinkPHP 还 文 持 多 种 目录 结构 的 部 署 ， 以 多 语言 配置 为 例 ，ThinkPHP 还 文 持 以 下 目录 


结构 。 
Lang 
common. E 公共 语言 包 
zh_cn.php wee 简体 中 文 语言 包 
zh_tw.php …………… 繁体 中 文 语言 包 


ThinkPHP 的 灵活 性 很 大 程度 就 体现 在 这 些 基础 的 配置 文件 上 ， 它 不 仅 支 持 多 种 部 敬 方 
式 ， 还 文 持 配置 文件 的 重 写 及 合并 。 在 实际 应 用 开发 中 ， 读 者 可 结合 官方 开发 手册 选择 适合 
的 目录 部 车 方式 (目录 方式 可 以 人 不同， 但 编程 是 一 样 的 ， 本 书 内 容 主 要 讲解 编程 知识 )。 


5.3 ”配置 文件 


前 面 的 内 容 已 经 涉及 配置 文件 创建 ， 在 此 不 再 重 述 ， 将 直接 讲解 配置 文件 参数 。 
ThinkPHP 通过 配置 文件 改变 项 目 属性 、URL 方式 及 运行 方式 等 ， 这 些 配置 信息 多 数 都 有 默 
认 的 参数 值 ， 开 友人 员 甚 至 不 需要 改变 配置 信息 ， 项 目 束 可 以 运行 。ThinkPHP 的 配置 文件 
使 用 标准 的 PHP 关联 数组 ， 通 过 键 值 对 的 方式 改变 配置 信息 ， 下 面 分 别 进行 介绍 。 


5.3.1 选项 配置 


config.php 是 项 目的 配置 文件 ， 存 放 于 项 目的 配置 文件 目录 中 ， 项 目 中 所 有 使 用 到 的 
属性 信息 均 可 以 在 config.php 文件 中 进行 配置 。 当 然 开 发 人 员 也 可 以 目 定 义 配 置 项 ， 在 获 
取 配 置 项 信息 时 ，ThinkPHP 提供 了 快捷 函数 C 进行 获取 ， 如 有 果 在 视 几 中 获取 ， 也 可 以 使 
用 {$Think.config. 配 置 项 名 称 } 标 签 进行 获取 。 一 个 简单 的 config.php 文件 参数 如 以 下 代码 
所 示 。 
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<?php 

return array 人 
// WEAN =>' 配置 值 ' 
'URL_CASE_INSENSITIVE' =>true, 


(DB TYPE" => 'mysql', // 数据 库 类 型 
DB HOST => 'localhost', // 服务 器 地 址 
'DB_NAME' => 'tp', // 数据 库 名 
'DB_USER' => 'root', // 用 户 名 
'DB_PWD' => 'root', // 密码 
'DB_PREFIX' => 'tpk_', // 数据 库 表 前 绥 


1: 


?> 


如 果 和 需要 江 加 日 定义 配置 参数 ， 只 需要 按照 格式 加 入 关联 数组 项 束 可 以 了 。 上 述 代码 为 
数据 库 连 接 配 置 参 数 ， 如 果 使 用 模块 分 组 方式 部 署 应 用 ， 应 该 放 到 全 局 配置 文件 中 
(Conf/config.php)。 因 为 通常 情况 下 数据 库 配 置信 息 都 是 全 局 的 ， 所 有 项 目 数 据 库 配置 信息 
都 是 一 样 的 。 但 是 ， 如 果 使 用 的 是 传统 的 项 目 部 普 方 式 ， 由 于 没有 全 局 配置 文件 的 概念 ， 所 
以 数据 库 配 置信 息 束 达 不 到 通用 的 目的 ， 需 要 在 各 目的 项 目 配置 文件 中 重复 定义 上 述 代码 ， 
这 在 大 型 的 应 用 开发 中 是 非 第 不 方便 的 ， 对 后 期 的 维护 来 讲 也 是 一 件 烦 琐 的 事 。 

好 在 ThinkPHP 使 用 数组 作为 配置 参数 ，PHP 本 身 对 数组 的 支持 是 非常 完善 的 ， 对 常见 
的 拆 分 、 重 组 、 排 序 等 都 提供 了 相应 处 理 函 数 。 在 此 只 需要 使 用 array_merge DÉI, DI ot zt 
现 数组 合并 ， 对 于 ThinkPHP 而 言 ， 无 论 合并 多 少 个 数组 ， 最 终 的 配置 都 是 后 面 的 参数 敢 产 
可 向 的 参数 。 

首先 在 网 站 的 根 目 录 创 建 一 个 config.inc.php 5 index.php 入 口 文 件 同 级 )， 该 文件 名 称 
可 以 目 定 义 ， 然 后 将 数据 库 配 置信 息 配 置 到 该 文件 中 ，config.inc.php 作为 全 局 文件 ， 网 站 中 
所 有 全 局 信息 应 该 都 在 该 文件 中 进行 配置 (如 网 站 名 称 、URL、 全 局 静态 资源 等 )。 

然后 打开 项 目 配 置 文 件 Conf/config.php)， 使 用 array_merse 子 数 合并 全 局 配置 文件 ， 如 
以 下 代码 所 示 。 


<?php 
Şarrl=include ("./config.inc.php");}; 


Sarr2=array ( 
"project_name"=>" 项 目 名 称 " 
'URL CASE _ INSENSITIVE' =>true 

) 7 

// 数 组 合并 

EE 

?> 


通过 以 上 步骤 ， 现 在 束 可 以 在 控制 喜 方 法 中 使 用 这 些 配置 信息 了 。 如 果 需 要 继续 在 其 他 
项 目 中 使 用 config.inc.php 配置 信息 ， 重 复 以 上 操作 即 可 。 和 常见 的 配置 参数 如 表 5-2 所 示 。 


表 5-2 config.php 常见 配置 参数 


URL. ROUTER ON 
URL_MODEL 


URL CASE INSENSITIVE URL 是 否 区 分 大 小 写 ， 建 议 设置 为 true 
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TMPL_DENY_PHP 
TMPL_L_DELIM 
TMPL_R_DELIM 
TAGLIB_BEGIN 
TAGLIB_END 
TMPL_CACHE_ON 
TMPL_CACHE_TIME 
LAYOUT_ON 
HTML_CACHE_ON 
SHOW_PAGE_TRACE 
TOKEN_ON 
APP_FILE_CASE 


DB_TYPE 

DB_HOST 
DB_NAME 
DB_USER 

DB_PWD 

DB_PORT 
DB_FIELDS_CACHE 


DB_CHARSET 


DB_DEPLOY_TYPE 
DB_RW_SEPARATE 
DB_MASTER_NUM 


DB_SQL_BUILD_CACHE 


TMPL_ACTION_SUCCESS 
TMPL_EXCEPTION_FILE 
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CZE) 
默 认 值 w "E 
flash 默认 视图 引擎 是 否 禁 用 PHP 代码 
默认 视图 引擎 标签 开始 标记 
默认 视图 引擎 标签 结束 标记 
标签 库 标 签 开始 标记 
标签 库 标 签 结 束 标记 
true 是 否 开 局 模板 编译 缓存 
模板 缓存 有 效 期 ， 只 有 TMPL_CACHE_ON 为 true 时 此 项 才 生 效 
false 是 否 开启 布局 功能 ，3.0 以 上 版 本 才 支 持 该 功能 
false 是 否 开局 静态 缓存 ， 可 参考 本 书 第 10 章 
false 显示 页 面 Trace 信息 ， 在 开发 阶段 应 该 开局 该 项 
true 视图 中 的 表单 是 否 生成 令 牌 ， 有 效 避 免 恶意 提交 
false 是 否 检查 文件 的 大 小 写 ， 在 Windows 平台 下 开发 时 ， 建 议 开启 
数据 库 配 置 选项 
mysql 数据 库 类 型 ， 可 选 的 参数 可 参考 ThinkPHP/Extend/Driver/Db 目录 


数据 库 地 址 

null 数据 库 名 称 

root 数据 库 登 录用 户 

null 数据 库 登 录 密 码 

null SO bat 

true 是 否 缓存 数据 表 字 段 

数据 库 编 码 ， 必 须要 与 真实 的 数据 库 编 码 相 对 应 ， 建 议 全 部 使 用 
utf8， 耕 则 有 可 能 将 出 现 乱码 

是 否 开启 数据 库 集群 ， 一 般 用 在 读 写 分 离 的 架构 上 ， 默 认 关 闭 。 


localhost 


utf8 


false 是 否 缓存 SQL 语句 

提示 设置 选项 
默认 成 功 跳 转 对 应 的 模板 文件 
think_exception.tpl | 异常 页 面 的 模板 文件 


dispatch_jump.tpl 


表 5-2 只 列 出 了 最 常见 的 配置 参数 ，ThinkPHP 提供 了 非常 多 的 配置 参数 ， 但 大 多 数 情 


DL PA ia E E AVE o 
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静态 缓存 配置 


ThinkPHP 提供 了 HTML CACHE ON 配置 参数 ， 用 于 设置 是 否 开 启 静 态 绥 存 功能 。 所 


谓 的 静态 绥 存 就 是 文件 缓存 (ThinkPHP 还 提供 了 内 存 缓 存 、 数 据 库 绥 存 多 种 方式 )， 
HTML CACHE RULES 关联 数组 可 以 灵活 地 配置 静态 绥 存 的 各 各 规则。 默认 情况 下 ， 系 统 
将 使 用 mds 的 方式 建立 缓存 ， 绥 存 规则 可 以 方便 地 创建 基于 控制 器 名 称 ， 动 作 名 称 的 缓存 


国 国 113 


PHP MVC 开发 实战 


文件 。 绥 存 规则 一 旦 配置 好 ， 系 统 会 目 动 识别 所 需要 绥 存 的 页 面 。 豆 态 规则 定义 在 绥 存 模式 
中 ， 第 见 的 缓存 模式 如 以 下 代码 所 示 。 


HTML_CACHE_RULES'=> array (' 
'ActionName'=>array (SR EN Il), ， 静态 缓存 有 效 期 !，“' 附 加 规则 ' ) ， 
'ModuleName '=>array (' 静 态 规 则 '，' 静 态 缓存 有 效 期 '，' 附加 规则 ' ) ， 
'ModuleName :ActionName'=>array (' 静 态 规则 ' ，' 带 态 缓 存 有 效 期 '，' 附加 规则 ')， 
'*'=>array (' 静 态 规则 ' ，' 融 态 缓 存 有 效 期 '，' 附加 规则 ' ) ， 


) 

FE, "gf 4 种 静态 绥 存 模式 。 其 中 ActionName 模式 表示 将 根据 所 有 控制 妖 
动作 方法) Gu SÉ Ce: ModuleName 将 根据 所 有 控制 右 建 立 绥 人 存 ; ModuleName: 
ActionName 将 根据 指定 的 控制 器 动作 建立 缓存 ，* 表 示 通 配 模式 ， 将 建立 全 站 静态 缓存 〈 所 
有 能 访问 的 页 面 )。 

缓存 有 效 期 和 附加 规则 都 是 可 选 参 数 ， 这 里 重点 需要 理解 静态 缓存 规则 。 议 态 缓 存 规 则 
将 约束 绥 存 文件 的 创建 方式 ， 以 下 代码 即 为 简单 的 静态 规则 。 

'HTMI，CACHE ON"=>trcue，/ /开启 静态 绥 存 

HTMIL CACHE TIME'=>3600, 

"HTML_PATH"=>" ./Html", // 设 置 静态 缓存 路 径 

“HTM te Te =- hno, 

'HTML_READ_TYPE' => 0, // Sie te DV rd 

'HTML_CACHE_RULES'=> array ( 

eonmeene. imde i array eont ent ee 


'index:index'=>array ('home/index'), 


) v 
如 以 上 代码 所 示 ， 一 共 定 义 了 两 条 静态 规则 。 其 中 第 1 条 将 Content 控制 器 中 的 Index 
动作 建立 静态 缓存 ， 静 态 缓存 文件 名 将 根据 Get 参数 进行 创建 ， 最 终 访 问 
http://tp.localhost/index.php/content/id/1 页 面 时 所 创建 的 静态 缓存 文件 名 为 content-1.html; 第 
2 条 将 根据 Index Prihlas index 动作 创建 静态 绥 存 ， 当 用 户 访问 http://tp.localhost 时 ， 系 统 自 
动 创建 首页 (index.html) 缓存 。 
默认 情况 下 静态 绥 存 文件 存放 在 home/Html 目录 下 ， 通 过 配置 参数 HTML _PATH 修改 
静态 绥 存 文件 存放 目录 。 需 要 注意 的 是 ， 开 月 静态 缓存 不 仅 需 要 正确 配置 
HTML_CACHE_RULES 选项 ， 还 必须 开启 HTML_CACHE_ON 选项 ， 两 者 缺 一 不 可 。 要 测 
试 静 态 绥 存 ， 需 要 注意 以 下 儿 个 步骤 。 
> 关闭 程序 的 调试 模式 ， 即 在 入 口 文件 中 设置 define("APP_DEBUG", false), TUIA 
关于 毅 态 缓存 的 选项 都 将 无 效 。 
> 如 朱 是 Linx 平台 需要 手动 在 项 目 根 目录 下 创建 一 个 名 为 Hm 的 目录 ， 然 后 赋 给 读 
写 权 限 。 
> 相应 的 动作 中 必须 要 有 输出 模板 〈 即 使 用 $this->displayO0)， 并 创建 相应 的 模板 文件 。 
> 在 非 调试 模式 下 ，ThinkPHP 会 生成 预 编 译文 件 Runtime/~runtime.php， 路 由 规则 修改 
后 需要 手动 删除 该 文件 。 
静态 绥 存 能 够 有 效 提 高 页 面 的 啊 应 速度 ， 尤 其 在 大 型 网 站 中 更 是 需要 使 用 静态 缓存 功 
能 ， 丁 傈 单 地 介绍 了 静态 绥 存 规则 配置 ， 更 详细 的 绥 存 功能 将 在 本 书 第 10 革 进 行 介 绍 ， 
在 此 读者 只 需要 理解 前 态 缓存 的 作用 即 可 。 
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5.3.3 ”路 由 配置 


路 由 是 ThinkPHP 为 了 改善 和 丰富 URL 呈现 方式 的 一 种 手段 ， 能 够 将 原本 元 长 难 记 的 
URL 变 得 形象 好 记 。URL_ ROUTE _ RULES 数组 用 于 配置 路 由 规则 ， 路 由 规则 的 正确 与 个 二 
接 决 定 了 URL 请 求 是 否 成 功 ， 所 以 理解 URL_ROUTE _ RULES 配置 选项 ， 是 使 用 ThinkPHP 
路 由 功能 的 基础 。 正 确 配置 URL_ ROUTE RULES 后 ， 必 须 开 启 URL_ROUTER_ON 方 可 起 
效 。 下 面 将 通过 一 个 简单 的 路 由 规则 简单 演示 URL 路 由 功能 ， 更 详细 的 路 由 使 用 方式 在 本 
E 6.1.2 节 将 有 人 介绍。 假设 现 在 需要 将 “http:/tp.localhosUindex.php/ 李 开 涌 ”路 由 到 “http:/ 
tp.localhost/index.php?a=content&m=index&name= ~F 7 ý”, HRA URL_ROUTE RULES 路 由 
规则 代码 如 下 所 示 。 

"URL_ROUTER_ON"=>true, 


"URL_ROUTE_RULES"=>array ( 
't/:name' =>'Index/content' // 路 由 到 Index 控制 器 下 的 content 动作 


), 
在 content 动作 中 可 以 使 用 $》_GET 获取 到 name 变量 的 值 ， 使 用 方式 和 标准 的 PHP 代码 
并 没有 区 别 ， 如 以 下 代码 所 示 。 
public function content () { 
echo E Yname™ ls 
} 
D 说 明 : ThinkPHP 2.x 在 缓存 及 路 由 配置 文件 部 署 上 有 所 不 同 。 静 态 缓存 及 路 由 配置 均 使 用 单独 的 配置 文 
件 实现 。 在 配置 规则 上 ， 与 ThinkPHP 3.x 则 保持 一 致 。 


5.4 小 结 


本 章 详细 地 介绍 了 ThinkPHP 文件 结构 。ThinkPHP 3.0 使 用 了 全 新 的 开发 模式 ， 其 中 了 豫 
动 与 扩展 是 全 新 架构 的 核心 设计 思念 ， 本 章 对 Lib 及 Extend 目录 进行 了 深入 介绍 ， 使 读者 能 
人 够 理解 整个 框架 的 文件 部 署 方式 。 

对 于 普通 的 PHP 开发 人 员 而 言 ， 虽 然 未 必需 要 理解 整个 框架 的 原理 ， 但 如 果 要 实现 定 
制 功 能 ， 这 些 都 是 必须 要 了 解 的 。 本 章 内 容 是 学 习 ThinkPHP MVC FRAN TR, HANZ 
都 是 基于 理论 知识 的 ， 从 下 一 曹 开始 将 以 本 章 内 容 为 基础 逐渐 进入 MVC 开发 的 实战 阶段 。 
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内 容 提 要 


ThinkPHP 是 一 款 轻 巧 型 的 MVC 开发 框架 ， 提 供 了 所 有 主流 框架 所 支持 的 特性 ， 开 发 
人 员 可 以 使 用 ThinkPHP 开发 出 强 半 而 又 高 效 的 PHP 应 用 。ThinkPHP 的 MVC 模式 吸收 了 
所 有 主流 框架 的 特点 ( 包括 Java、ASPNET ) 提供 了 层次 分 明 、 设 计 合 理 的 MVC 分 层 设 
计 。 针 对 国内 情况 提供 了 4 种 URL 模式 ， 有 效 改善 了 用 户 使 用 体验 。 

本 章 首先 介绍 URL 模式 。 无 论 在 哪个 框架 ，URL 模式 都 是 框架 的 重要 内 容 ， 读 者 需要 
深入 掌握 ; 最 后 将 介绍 MVC 开发 的 详细 步 又 ， 通 过 本 章 学 习 读 者 将 会 彻底 理解 三 大 部 件 
(Model、View、Controller ) 在 ThinkPHP 中 的 使 用 。 


学 习 目 标 


了 解 ThinkPHP 处 理 URL 的 方式 。 
掌握 ThinkPHP 控制 句 的 使 用 。 

掌握 ThinkPHP 模型 器 的 使 用 。 

掌握 ThinkPHP 视图 引擎 的 实战 应 用 。 
加 深 对 MVC 各 部 件 的 认识 。 
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6.1 ThinkPHP 中 的 URL 


6.1.1 URL 模式 


ThinkPHP 共 文 持 4 种 形式 的 URL Bd, 分别 为 普通 人 模式、PATHINFO 模式 、 
REWRITE Bd Him, X 4 Ph URL 构成 了 ThinkPHP MVC 任务 请 求 的 核心 ， 开 发 人 
员 可 以 通过 配置 文件 进行 指定 ，4 种 URL 模式 根据 Web IRIRE LRAD A 
些 服务 器 如 Nginx, lighttpd 需要 读者 上 自行 配置 )， 从 目前 的 使 用 情况 来 看 ， 在 Apache 中 运行 
得 最 完美 ， 开 发 人 员 不 需要 特别 的 设置 均 可 以 在 4 种 URL 中 目 由 切换 。 下 面 分 别 对 4 种 
URL 进行 讲解 。 

1. 普通 模式 

本 质 上 ThinkPHP 的 4 种 URL 都 是 由 普通 模式 的 URL 转换 而 成 的 ， 普 通 模式 是 整个 
URL 处 理 模 式 的 基础 。 普 通 模式 下 的 URL 显得 比较 长 ， 各 参数 需 显 式 传递 ， 过 长 的 URL 
对 搜索 引擎 并 不 友好 ， 但 是 普通 模式 能 够 文 持 任 何 Web 服务 器 ， 对 开发 人 员 而 言 也 显得 比 
较 友 好 。 普 通 模式 下 URL 格式 如 下 。 

http://tp.localhost/index.php?a=index&m=index&username=ceiba 

普通 模式 URL F LAIN URL 非 利 类 似 ， 它 是 以 显 式 的 参数 作为 传递 方式 。 其 中 参 
数 a 表示 控制 右 的 动作 《 即 模型 关中 的 方法 ) 参数 m 指定 控制 费 名 称 。 除 了 需要 指定 这 两 
个 参数 之 外 ， 开 发 人 员 可 以 附加 更 多 上 日 定义 参数 ， 附 加 的 日 定义 参数 在 控制 右 动 作 中 可 以 使 
用 $_G[“ 参 数 名 称 ”] 方 式 进 行 获 取 。 如 琳 请 求 的 控制 桌 和 动作 都 为 index， 那 么 可 以 省 略 该 请 
求 参 数 ， 格 式 如 下 。 

http://tp.localhost/index.php?username=ceiba 

EX URL 的 请 求 结 末 是 和 前 面 URL kä Ern. Zem URL 模式 不 需要 修改 
Web 服务 器 配置 ， 只 需要 修改 配置 文件 参数 URL_MODEL=0 即 可 ; 普通 的 URL 模式 是 早期 
ThinkPHP 的 默认 URL 便 式 ， 也 是 众多 网 站 所 使 用 的 方式 ， 但 无 论 对 用 户 还 是 搜索 引擎 都 显 
得 不 友好 ， 所 以 现在 很 少 使 用 了 。 接 下 来 将 介绍 另外 一 种 比较 新 式 的 URL d. 

2. PATHINFO 模式 

PATHINFO 模式 能 够 提供 一 种 简短 、 友 好 的 URL 形式 ， 是 ThinkPHP 默认 的 URL $ 
式 。PATHINFO 模式 是 利用 ThinkPHP 的 路 由 功能 实现 的 ， 它 的 实现 原理 与 URLRewriter 相 
类 似 ， 但 是 ThinkPHP 不 依赖 于 Web 服务 器 的 正则 ， 框 架 上 自身 已经 有 一 僚 URL 正则 处 理 机 
制 ， 开 友人 员 甚 至 不 需要 部 赣 .htaccess 文件 ， 所 以 PATHINFO 能 够 兼容 绝 大 多 数 Web 服务 
28. PATHINFO 模式 的 URL 格式 如 下 。 

http://tp.localhost/index.php/index/content/username/ceiba 

PATHINFO 为 ThinkPHP SAUA MADIER, PEE wN URL 形式 ， 提 高 用 户 体验 。 
PATHINFO 清楚 地 表达 MVC 的 处 理 过 程 ， 如 图 6-1 ra, 

如 图 中 所 示 ， 在 整个 请 求 的 URL 字符 串 中 ， 对 于 开发 者 而 言 只 需要 记 住 两 项 参数 ， 即 
Piar Gndex) 和 content 动作 )， 前 后 顺序 不 能 改变 。 其 余 的 参数 (选项) 均 可 作为 需要 


国 国 117 


PHP MVC 开发 实战 


传递 的 GET 参数 ， 如 username/ceiba 与 &username=ceiba 作用 是 一 样 的 ， 如 果 需 要 传递 更 多 
参数 ， 依 次 类 推 ， 附 加 上 即 可 。 


http://tp.localhost/index.php/index/content/username/ceiba 


Geer? 


图 6-1 PATHINFO 模式 


入 口 网 址 后 的 参数 分 隔 符 是 可 变 的 ， 开 发 人 员 可 以 通过 修改 URL_PATHINFO_DEPR 配 
置 参数 实现 需求 ， 默 认 URL _PATHINFO_DEPR=”*”， 常 见 的 分 隔 符 为 “”、“,”“-” 等 ， 
但 不 能 使 用 “:”“&”“?3” 作 为 分 隔 符 。 

PATHINFO 的 本 质 是 使 用 PHP 解释 引擎 中 的 $_SERVER[PATH_INFO] 数 组 实现 的 ， 能 
不 能 使 用 PATHINEFO 模式 还 得 看 服务 器 文 不 文 持 $_SERVER['PATH_INFO]， 通 常情 况 下 党 
见 的 Web 服务 器 如 Apache, US 等 都 能 够 很 好 地 支持 。 如 果 是 反问 代理 一 类 的 Web 服务 器 
(例如 Nginx、lighttpd)， 需 要 额外 调用 CGI 详细 可 阅读 本 书 附 录 A). 

PATHINFO 配合 服务 右 的 RewriteRule 功能 ， 还 可 实现 隐 蕊 index.php。 在 Apache 中 只 
需要 开启 mod_rewrite WBA] (XAMPP 开发 套件 已 默认 开局 )， 然 后 配置 网 站 中 的 .hessace 
文件 (与 入 口 文 件 同 级 )， 如 以 下 代码 所 示 。 


<IfModule mod rewrite.c> 


RewriteEngine on 


RewriteCond %{REQUEST_FILENAME} !-d 

RewriteCond %{REQUEST_FILENAME} !-f 

RewriteRule ^(.*)$ index.php/$1 [QSA,PT,L] 
</IfModule> 


重启 Apache k3 4s. JETI nJ LEM http://tp.localhost/index/content/username/ 
ceiba {t$ http://tp.localhost/index.php/index/content/username/ceiba Wht Y o KEJ index.php 
文件 后 ， 不 仅 让 URL 更 加 友好 和 简洁 ， 同 时 也 让 URL 更 加 通用 。 


O 提示 : Ajax、SOAP、 移 动 终端 请 求 等 许多 应 用 是 不 文 持 PATHINFO 模式 的 ， 甚 至 不 支持 普通 的 模式 ， 
这 时 使 用 隐藏 index.php 文件 后 的 PATHINFO 模式 ， 能 够 让 URL 被 任何 类 型 的 应 用 调用 〈 因 为 会 被 解 
释 成 目录 ) o 


3. REWRITE 模式 

REWRITE 模式 就 是 URL 重 写 规则 ， 它 能 够 完全 实现 PATHINFO 模式 的 所 有 功能 ， 还 
能 够 利用 服务 器 的 URL 重 与 功能， 实现 更 完善 的 URL 模式 。REWRITE 不 使 用 PHP 的 
PATCHINO 功能 ， 能 不 能 文 持 REWRITE 模式 关键 要 看 服务 占 文 不 文 持 RewriteRule 功能 
( 绝 大 多 数 都 是 支持 的 )。REWRITE 模式 需要 读者 额外 熟悉 Web 服务 器 的 各 项 配置 ， 这 不 是 
本 书 的 重点 。 以 下 .htaccess 文件 代码 为 笔者 备用 的 REWRITE 配置 ， 它 能 够 模拟 基本 的 
PATHINEFO。 


<IfModule mod rewrite.c> 


RewriteEngine on 
RewriteCond %{REQUEST FILENAME} !-d 
RewriteCond %{REQUEST FILENAME} !-f 
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RewriteRule ^(.*)$ index.php/$1 [QSA,PT,L] 
</IfModule> 


4. 兼容 模式 

普通 模式 能 够 保证 网 站 在 所 有 支持 PHP 的 Web 服务 器 下 正常 运行 ， 但 是 由 于 用 户 体验 
差 ， 对 搜索 引擎 不 友好 已 经 很 少 被 采用 了 ; 而 PATHINFO 模式 虽然 有 效 地 改善 了 URL JÉ 
式 ， 提 高 了 用 户 体 验 ， 是 所 有 MVC 框架 所 采用 的 主流 方式 ， 但 由 于 受 限 于 一 些 服 务 器 ， 
PATHINFO 并 不 能 完美 地 让 项 目 适 用 于 任意 场景 ， 所 以 ThinkPHP 提供 了 URL RAI. AR 
容 模 式 能 够 让 普通 模式 与 PATHINFO 模式 互相 切换 ， 对 开发 者 而 言 整个 项 目 不 需 要 修改 代 
码 ，ThinkPHP 可 以 智能 地 处 理 善 通 模式 与 PATHINFO 模式 之 间 的 切换 。 兼 容 模式 的 URL 表 
现形 式 如 下 所 示 。 

http://tp.localhost/index.php/?s=/index/content/username/ceiba/ 

RARAN S RAA PATHINFO 的 特点 ， 在 表现 形式 上 是 一 种 折 中 的 方案 ， 并 
不 是 好 的 MVC URL 模式 ， 它 的 最 大 好 处 束 是 能 够 确保 网 站 运行 在 任何 Web 服务 器 。 

事实 上 兼容 模式 本 质 上 还 是 普通 模式 ， 它 由 参数 s 传递 MVC 所 需要 的 参数 〈s 参数 后 
的 分 隔 符 由 URL_PATHINFO_DEPR 配置 项 指定 )， 一 般 用 于 开发 阶段 的 调试 。 要 局 用 兼容 
模式 ， 只 需要 修改 配置 项 'URL_MODEL=>3 即 可 。 

EX 4 种 URL 模式 读者 可 根据 需要 进行 选择 ， 但 无 论 是 从 用 户 体 验 还 是 对 搜索 引 苟 友 
好 考虑 ， 笔 者 都 建议 使 用 PATHINFO 或 REWRITE 模式 。 如 果 需 要 做 成 全 兼容 的 网 站 〈 如 供 
第 三 方 调用 的 接口 、SOAP 等 ) 还 需要 隐藏 入 口 文件 〈 当 然 接 来 下 要 介绍 的 URL 路 由 也 是 
一 种 解决 方案 )， 为 了 便于 讲解 ， 统 一 规范 ， 如 无 额外 说 明 ， 本 书后 面 的 所 有 内 容 都 将 使 用 
PATHINFO 作为 URL 请 求 模式 。 


6.1.2 URL 路 由 


使 用 MVC 的 好 处 之 一 束 是 能 够 提供 可 定制 化 的 URL 形式 ， 其 中 路 由 功能 是 MVC 框架 
中 最 常见 的 功能 。ThinkPHP 提供 了 完善 的 URL 路 由 功能 ， 能 够 实现 高 度 可 定制 的 URL 形 
式 。 在 ThinkPHP 3.0 F, URL 路 由 规则 使 用 URL_ROUTE_RULES 数组 进行 配置 ， 而 在 
ThinkPHP 2.x 中 使 用 的 是 routes.php 配置 文件 ， 这 点 需要 注意 。 下 面 为 了 便于 讲解 ， 将 以 URL 
“http://tp.localhost/index.php/t/ 李 开 涌 ”路 由 人 到 “http://tp.localhost/index.php?a=content&m= 
index&name= 李 开 涌 &email=kf@8605SS.com” 为 示例 ， 详 细 介绍 ThinkPHP 提供 的 几 种 路 由 
规则 形式 《所 有 路 由 规则 都 是 基于 项 目 不 分 组 的 ， 即 多 入 口 文件 模式 )。 

1. 键 对 值 形式 

格式 : 路 由 规则 =>'[ 模 块 /操作 ]? 额 外 参数 1= 值 1& 额 外 参数 2= 值 2... 


"URL_ROUTE_RULES"=>array ( 
't/:name' =>'Index/content?email=kf@86055.com' 


) ， 

键 值 对 形式 的 路 由 规则 容量 比较 小 ， 它 是 以 关联 数据 作为 配置 规则 。 其 中 t/:name 表示 
路 由 规则 ， 路 由 规则 决定 了 URL 表现 的 形式 ; Index/content 表示 路 由 的 目标 ，Index KRPE 
HAZE, content 表示 控制 右 动 作 ， 顺 序 不 能 相反 。t 后 的 分 隅 符 “/” 表 示 一 个 路 由 匹配 的 
结束 ， 它 与 Index/content 进行 呼应 。 
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2. 键 对 数组 形式 
格式 : ' 路 由 规则 =>array([ 模 块 /操作 ], 人 额外 参数 1= 值 1& 额 外 参数 2= 值 2...) 
"URL_ROUTE_RULES"=>array ( 

't/:name' =>array ("Index/content","email=kf@86055.com"), 


), 

键 对 数组 形式 和 键 对 值 形式 是 类 似 的 ， 不 同 的 是 路 由 目标 的 形式 使 用 的 一 个 关联 数组 。 
可 见 ， 键 对 数组 形式 无 论 是 格式 上 还 是 容量 上 都 比 单 一 的 键 对 值 形式 要 灵活 。 

3. 跳 转 形式 

格式 : 路 由 规则 =>' 处 部 地 址 ' 

0 

跳 转 形式 的 路 由 规则 形式 就 是 一 个 页 面 跳 转 ， 在 实际 应 用 开发 中 的 意义 不 大 。 外 部 地 址 
是 一 个 字符 串 ， 可 以 是 任何 有 效 的 URL 地 址 。 

4. 重 定向 跳 转 

格式 : ' 路 由 规则 =>array( 外 部 地 址 ', 重 定 问 代 伍 ") 

有 

&email=kf@86055.com', 'HTTP/1.1 301 Moved Permanent Lwvi) 


重 定 问 跳 转 是 普通 跳 转 模式 的 增强 模式 ， 它 在 跳 转 到 外 (内 )〉 部 地 址 时 还 会 回 浏 览 右 头 
部 (header) 输出 额外 信息 。 例 如 搜索 引擎 对 于 普通 的 跳 转 是 不 会 索引 的 ， 但 在 跳 转 时 给 浏 
览 器 的 头 部 附加 上 “HTTP/1.1 301 Moved Permanently” 人 信息， 此 时 搜索 引擎 就 认为 该 链接 是 
一 条 正常 的 数据 链接 ， 会 投 正 第 的 程序 进行 索引 。 第 用 的 浏览 句 头 部 信息 有 301、302 等 。 

上 述 4 种 路 由 形式 中 ， 主 要 使 用 的 是 第 1 和 第 2 种 。 这 里 还 需要 深入 理解 路 由 规则 ， 所 
谓 的 路 由 规则 就 是 与 浏览 器 相配 对 的 参数 ， 在 浏览 器 中 呈现 的 参数 可 以 有 若干 个 ， 但 一 条 路 
由 规则 只 对 应 一 个 参数 。 

在 没有 使 用 路 由 之 前 的 传统 MVC 控制 流程 中 ， 无 论 哪 种 URL 模式 都 必须 要 显 式 地 给 
出 控制 器 名 称 和 动作 名 称 (index 除外 ，PATHINFO 使 用 参数 位 置 表示 )。 但 使 用 URL 路 由 
之 后 ， 控 制 器 和 控制 器 动作 都 可 以 取消 了 ， 开 发 人 员 可 以 使 用 任何 一 个 字符 (包括 中 文 ) 代 
乱 探 制 器 和 动作 名 称 ， 这 在 主流 的 大 型 网 站 中 是 非常 多 见 的 ， 例 如 狐 浪 微 博 、facebook 等 。 

URL 路 由 并 不 一 定 需要 MVC 框架 才能 实现 ， 事 实 上 使 用 普通 的 PHP 来 做 URL 路 由 在 
超大 并 发 的 环境 下 也 不 是 很 理想 。URL 路 由 完全 可 以 使 用 Web 服务 器 的 重 写 功能 来 实现 ， 
但 无 论 是 复杂 度 还 是 灵活 性 都 不 是 一 般 的 PHP 程序 员 所 能 接受 的 ，ThinkPHP 提供 的 路 由 处 
理 机 制 简化 了 路 由 配置 流程 ， 降 低 了 门槛 ， 使 得 网 站 应 用 开发 人 员 更 专注 于 业务 逻辑 。 

现在 再 回 过 头 来 看 前 面 的 路 由 规则 形式 。“t/:name” 是 一 种 动态 写法 ,“:” 表 示 的 是 一 
个 可 接收 的 变量 ， 如 果 不 使 用 “:” 则 不 是 变量 ， 而 是 一 个 固定 的 字符 串 ;“name” 表 示 变 量 
名 称 ， 它 与 URL 字符 串 “t” 后 面 的 参数 进行 配对 。 变 量 是 可 以 加 限制 符 的 ， 限 制 符 以 正则 
表示 陈 中 的 匹配 符号 作为 条 件 ， 例 如 限制 变量 name 只 能 接收 纯 数字 ， 代 但 如 下 。 


"URL_ROUTE_RULES"=>array ( 
't/:name\d' =>array ("Index/content","email=kf@86055.com"), 


L 
加 入 限制 符 以 后 ， 在 content 动作 中 使 用 $_G[“name”] 进 行 接收 ， 就 再 也 获取 不 到 传 参 值 
了 ， 因 为 “ 李 开 涌 ”并 非 为 数字 。 读 者 可 以 尝试 改 成 数字 进行 验证 。 
路 由 规则 不 仅 能 够 使 用 限制 字符 ， 还 能 够 使 用 排除 字符 。 同 样 ， 排 除 字 符 使 用 “^” 符 


120 D EI 


第 6 章 ThinkPHP 开发 MVC 应 用 


号 进行 定义 ， 多 个 排除 字符 使 用 “|” 隔 开 ， 如 以 下 代码 所 示 。 
"URL_ROUTE_RULES"=>array ( 
ENEE e i mL on 


), 

如 上 述 代码 所 示 ， 使 用 排除 学 符 之 后 ， 在 浏览 右 中 输入 “t/add”、“edit”“delete” 了 字符 
都 将 被 过 滤 ， 在 content 动作 中 将 不 能 接收 到 这 些 参 数值 。 

在 以 上 几 个 示例 中 ， 路 由 名 称 “t” 都 是 固定 的 ， 它 必须 要 与 浏览 器 地 址 中 的 t 进行 对 
应 。 使 用 Web 服务 器 URL 重 写 大 多 数 都 是 使 用 正则 来 实现 路 由 功能 的 。 同 样 ， 在 
ThinkPHP 中 路 由 名 称 也 是 文 持 正则 匹配 的 ， 使 用 正则 匹配 后 显得 更 加 有 灵活， 只 要 URL 请 
求 字 符 串 符合 定义 的 正则 ，ThinkPHP 就 可 以 将 相应 的 地 址 路 由 到 目标 地 址 中 ， 如 以 下 代码 
所 示 。 


"URL_ROUTE_RULES"=>array ( 
'^/t\/.*?$/' =>array ("Index/content","email=kf@86055.com"), 


大 

使 用 正则 后 ， 控 制 器 、 动 作 、 参 数 都 可 以 在 一 条 正则 中 进行 配对 。 使 用 正则 表达 式 ， 可 
以 实现 任何 形式 的 URL， 它 的 灵活 性 是 最 高 的 ， 但 需要 读者 额外 熟悉 正则 表达 式 (正则 表达 
式 并 非 PHP 特有 的 功能 ， 它 是 一 套 规 范 ， 在 JavaScript 等 许多 脚本 中 应 用 得 很 广泛 ，MVC 
框架 本 身 就 是 利用 了 大 量 正 则 来 进行 MVC 控制 的 )。 


6.1.3 ”自动 生成 匹配 的 URL 


ThinkPHP 提供 了 4 种 URL 模式 ， 在 实际 应 用 开发 中 也 许 会 使 用 多 种 模式 进行 切换 。 这 
样 一 来 就 必须 面 对 一 个 问题 ， 假 设 开发 者 使 用 PATHINFO 作为 项 目的 URL 模式 ， 但 是 在 真 
实 的 生产 环境 中 发 现 服务 器 并 不 支持 PATHINFO， 这 时 原先 在 模板 中 写 好 的 链接 〈 例 如 静态 
资源 链接 、 栏 目 乙 间 的 导航 链接 等 ) 将 找 不 到 相应 的 资源 ， 只 能 重新 改动 模板 ， 工 作 量 是 非 
常 大 的 。ThinkPHP 提供 TU 函数 ， 可 以 让 URL 在 4 种 模式 中 目 由 切换 ， 而 模板 并 不 需要 改动 
任何 代 但 。 

U 函数 的 格式 为 : U([ 分 组 /控制 器 /操作 ]? 参 数 ' [参数 ', 伪 静态 后 缀 ', 是否 跳 转 ', 显 示 
域名 '])， 其 中 分 组 指 的 是 项 目 分 组 ， 如 果 不 使 用 项 目 分 组 则 不 需要 填写 ; ? 后 面 的 字符 串 
是 需要 生成 的 参数 ， 为 了 代码 简洁 ， 可 以 使 用 数组 存放 。U 函数 文 持 URL 路 由 ， 下 面 将 分 
列 介 绍 。 

假设 我 们 需要 让 “http://tp.localhost/index.php/index/content/username/ceiba”URL 能 够 在 
4 种 URL 模式 下 运行 ， 那 么 可 以 使 用 函数 U 生成 ， 如 下 所 未 。 

U("Index/content?username=ceiba") 

参数 “username” 可 以 使 用 数组 存放 ， 如 下 所 示 。 

U( Index/content ,array( "Username => "celba )); 

和 其 他 配置 文件 一 样 ， 配 置 URL 参数 也 是 使 用 关联 数组 ， 如 末 有 多 对 参数 需要 生成 ， 
则 填写 相应 的 数组 项 即 可 。 数 组 中 的 键 值 对 和 URL 字符 串 中 的 参数 与 参数 值 是 相对 应 的 ， 
最 终 上 述 URL 在 4 种 URL 模式 中 的 表现 形式 如 下 。 

Gd) 普通 模式 


http://tp.localhost/index.php?m=index&a=content&username=ceiba 
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(2) PATHINEFO 模式 

http://tp.localhost/index.php/index/content/username/ceiba 

(3) REWRITE 模式 

http://tp.localhost/index/content/username/ceiba 

或 者 

http://tp.localhost/index.php/index/content/username_ceiba.html 

(4) IERTE 

http://tp.localhost/index.php?s=/content/username/ceiba 

在 生成 URL 参数 时 ， 可 以 使 用 数据 库 中 的 信息 作为 参数 值 。 关 于 数据 库 的 使 用 方式 将 
在 第 7 章 进 行 介绍 ， 在 此 读者 只 需要 了 解 即 可 。 生 成 URL 后 ， 界 面 设 计 人 员 可 以 在 视图 中 
进行 获取 了 《为 了 便于 党 理 ， 建 议 将 URL 统一 生成 一 个 数组 ， 在 视图 中 使 用 数组 下 标的 方 
式 获 取 )， 如 以 下 代码 所 示 。 


public function index (){ 


$url["index_content url"]=U("Index/content",array ("username"=>"ceiba")); 
Ssūürl[“index user elt leet 


SE ee DE eeler 


} 

同样 ，U AARE EELT URL 路 由 的 生成 ， 但 在 实际 应 用 开发 中 该 功能 使 用 得 比较 
少 ， 因 为 URL 路 由 严格 意义 上 来 讲 只 是 一 个 URL 的 别名 ， 它 的 表现 形式 都 已 经 固定 不 需要 
切换 ， 没 必要 使 用 PHP 代码 生成 。 使 用 函数 生成 URL 的 格式 如 下 。 

路 由 规则 : mews/:idld=>'News/read' 生成 格式 : U ('/news/1') 

URL 生成 不 是 必需 的 ， 它 的 功能 只 是 为 了 让 URL 能 够 在 4 种 URL 模式 中 更 容易 
切换 ， 但 事实 上 在 真实 的 项 目 开发 中 ， 生 产 环境 都 是 固定 的 ， 选 定 一 种 URL 模式 后 较 
少 改动 。 


6.1.4 ZIHARA 


REWRITE A DIRO TEH FEITES AERE, EH U RZ AAEE AH 
应 的 URL， 但 由 于 需要 按照 ThinkPHP 内 置 的 规则 进行 生成 ， 所 以 灵活 性 受到 一 些 限制 。 如 
果 项 目 已 经 确定 了 一 种 URL 形式 ， 那 么 完全 可 以 抛 开 ThinkPHP 的 URL 模式 ， 使 用 Web JR 
务 喜 的 重 写 功能 实现 更 简单 的 定制 需求 。 例 如 新 闻 网 站 多 数 都 是 静态 化 的 ， 在 ThinkPHP 中 
可 以 使 用 静态 缓存 来 达到 静态 化 的 目的 ， 然 后 配置 URL 重 写 规则 功能 即 可 ， 如 以 下 .htaccess 
文件 代码 如 示 。 


<IfModule mod rewrite.c> 


RewriteEngine on 

RewriteCond %{REQUEST_ FILENAME} !-d 

RewriteCond %{REQUEST FILENAME} !-f 

RewriteRule ^(.*)$ index.php/$1 [QSA,PT,L] 

RewriteRule ([p]{1,})/([a-zA-Zz,0-9]{1,})\.html$ content/index/id/$2 

RewriteRule ([c]{1,})/([a-zA-zZz,0-9]{1,})\.html$ column/index/id/$2 

RewvciceRule (lell 1) /(la=za-4, 09] e 


RewriteRule ([u]{1,})-([a-zA-Z,0-9]{1,})$ user.php/UserOperation/WeiBo_Login/ac/$2 
RewriteRule ([u]{1,})-([a-zA-Z,0-9]{1,})/$ user.php/UserOperation/WeiBo Login/ac/$2 
</IfModule> 
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配置 完成 URL 重 写 规则 之 后 ， 可 以 直接 在 视图 中 使 用 重 写 后 的 URL。 使 用 服务 器 的 
URL 重 写 功能 可 以 很 好 地 实现 URL PERM, A Varnish, Squid 等 缓存 服务 器 ， 能 够 实现 
性 能 更 高 的 PHP 动态 网 站 ， 这 在 超大 型 网 站 中 是 经 党 使 用 的 技术 方案 。 


6.2 ”模型 ( Model ) 


在 ThinkPHP 中 模型 共有 两 种 。 一 种 指 的 是 Model 基 类 本 喘 ， 称 为 数据 库 模 型 类 ; 另 一 
种 指 的 是 继承 于 Model 基 类 的 日 定义 类 ， 称 为 日 定 义 模 型 。 两 种 模型 可 以 混合 使 用 ， 但 数据 
库 模 型 也 可 以 且 接 在 控制 费 中 调用 ， 不 需要 目 定 义 模 型。 而 目 定义 模型 通常 是 为 了 增强 数据 
库 模 型 功能 (也 可 当成 普通 的 功能 类 使 用 )， 由 于 数据 库 模型 涉及 较 多 的 内 容 ， 本 书 将 使 用 
一 革 的 篇 幅 进 行 讲解 ， 下 面 首先 对 目 定 义 模 型 进行 介绍 。 


6.2.1 创建 模型 


关于 模型 的 细节 在 前 面 的 内 容 中 已 丝 有 过 介绍 ， 在 此 束 不 重复 叙述 。ThinkPHP 的 基础 
模型 类 名 为 Model， 该 类 定义 了 操作 数据 库 的 常用 方法 ， 继 承 于 Model 类 的 PHP KIN B 
定义 模型 。 在 其 他 一 些 MVC 框架 中 ， 模 型 被 映射 成 一 张 数据 表 ， 但 在 ThinkPHP 中 模型 并 
非 一 定 要 上 映射 为 一 张 数 据 表 ， 它 更 像 是 控制 豆 的 中 间 件 ， 这 层 中 间 件 能 够 操作 数据 库 、 读 写 
文件 、 数 据 转换 等 。ThinkPHP 对 模型 的 定义 不 太 严 格 ， 对 中 小 型 的 团队 开发 而 言 ， 这 种 
MVC 处 理 方 式 是 高 效 和 灵活 的 ， 开 发 人 员 可 以 像 使 用 PHP 普通 类 一 样 调用 模型 ， 下 面 将 介 
绍 怎样 创建 一 个 正确 的 目 定义 模型 类 。 

在 home/Lib/Model 目录 下 创建 一 个 PHP 文件 ， 并 命名 为 UserModel.class.php， 打 开 该 
文件 ， 创 建 一 个 PHP 类 ， 命 名 为 UserModel ， 并 使 该 类 继承 于 Model 类 。 这 样 ， 
UserModel.class.php 就 是 一 个 标准 的 ThinkPHP 项 目 模型 文件 。 为 了 便于 讲解 ， 现 在 需要 让 
该 模型 返回 一 些 数据 ， 代 人 码 如 下 。 


<?php 


class UserModel extends Model{ 
/** 
* 提供 数据 
= Enter CESCELOELON bere, 
E 
protected function userData(){ 
return array 人 
"1"=>array ( 
"姓名 "=>" 李 开 涌 "， 
ol 
"邮箱 "=>"kf@86055.com", 
"爱好 "=>" 读 书 " 
), 
"2"=>array ( 
"姓名 "=>" 李 明 "， 
"HER "=>" E", 
"邮箱 "=>"ceiba 126.com", 
"=> ln 
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} 
/大 大 
* 返回 数据 
s Enter description bere ooo 
* Qparam unknown type Susertd 
a 
public function infoData (Ş$userId=null){ 
if (empty ($userId)){ 
return $this->userData(); 
}else{ 
Şarray=Ş$this->userData(); 
return $array [Ş$userId]; 


} 


?> 


上 述 类 文件 代码 中 共有 两 种 方法 。userData 方法 用 于 存放 用 户 数据 ， 在 实际 应 用 开发 
中 这 些 数 据 可 以 在 数据 库 、NOSQL 或 者 文本 文件 中 获取 ， 这 里 为 了 方便 讲解 ， 只 使 用 
PHP 数组 存放 用 户 数 据 ; infoData 方法 将 根据 参数 userId 返回 相应 的 用 户 数据 。 其 中 
infoData 使 用 了 public 修饰 他 ， 表 示 该 方法 是 公开 可 调用 的 ， 而 userData 方法 使 用 
protected 修饰 符 ， 限 制 了 该 方法 只 能 在 本 模型 内 使 用 。 通 过 前 面 的 介绍 ， 可 见 模 型 与 常见 
的 PHP 类 非常 类 似 ， 如 果 读 者 不 习惯 模型 这 一 称呼 ， 不 妨 暂 时 可 以 理解 为 项 目 类 。 虽 然 如 
此 ， 模 型 与 标准 的 PH 类 还 是 有 些 不同 的 ， 在 一 些 应 用 场合 下 两 者 不 能 混为一谈 ， 在 目 定 
义 模 型 时 需要 注意 以 下 事项 。 
> 模型 的 文件 命名 规则 必须 要 严格 按照 “模型 名 +Model+.class.php” 的 方式 进行 命名 。 
> 模型 名 称 首 字 母 必 须 为 大 写 ， 后 绥 名 称 .class.php 不 能 写成 .php。 
> 模型 类 必须 继承 于 Model (或 者 AdModel) 类 ， 访 类 是 一 个 实例 类， 目 定 义 的 模型 
类 也 必须 为 实例 类 〈 不 能 定义 为 抽象 类 、 静 态 类 等 )。 
> 虽然 在 ThinkPHP 中 模型 未 必要 与 数据 库 中 的 数据 表 进 行 映射 ， 但 事实 上 模型 都 是 用 
于 处 理 数 据 库 信 息 的 ， 所 以 应 该 让 模型 名 与 数据 表 名 相对 应 。 
> 项 目 中 目 定义 的 模型 应 尽量 避免 互相 继承 。 
> 模型 类 不 能 使 用 构造 水 数 。 
> 模型 类 方法 尽量 避免 使 用 静态 方法 。 


6.2.2 ”实例 化 模型 


模型 本 质 上 是 一 个 实例 类 ， 该 类 封装 了 操作 数据 库 的 第 见方 法 ， 模 型 一 旦 被 实例 化 ， 意 
味 着 开始 链接 数据 库 。 前 面 已 经 介绍 过 ，ThinkPHP 不 强制 自 定 义 模 型 与 数据 表 进 行 映射 ， 
开发 人 员 完 全 可 以 当成 一 个 普通 的 类 文件 来 使 用 ， 所 以 使 用 new 方式 也 是 没 任何 问题 的 。 为 
了 简单 操作 ，ThinkPHP 还 提供 了 3 种 方式 快速 实例 化 模型 ， 如 下 所 示 。 

> new Model(' 数 据 表 名 ')， 使 用 传统 的 new 方式 实例 化 Model 基 类 ， 访 类 封装 了 操作 

数据 库 的 常见 方法 。 需 要 注意 的 是 new 方式 不 能 实例 化 自 定义 的 模型 。 

> Mmame=",class='"Model)， 快 捷 函 数 实例 化 模型 。 参 数 name 表示 数据 表 名 称 ，class 
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表示 上 自 定 模 型 类 。M 函数 可 以 在 初始 化 数据 表 后 再 实例 化 一 个 日 定义 的 模型 类 。 
> Damame=",app=")，D 函数 专门 用 于 实例 化 目 定 义 模型 ， 人 参数 name 表示 模型 名 (不 需 
要 Model.class.php); app 表示 项 目 名 称 ， 如 果 调 用 路 项 目的 目 定 义 模 型 ， 那 么 该 参数 
是 必需 的 ， 为 空 时 则 表示 当前 项 目 。 
关于 怎样 使 用 Model 基 类 进行 数据 库 操作 ， 在 本 书 第 7 章 将 有 深入 的 介绍 ， 在 此 读者 只 
需要 理解 该 类 的 作用 即 可 。 继 续 以 前 面 的 内 容 为 例 ， 如 果 现 在 需要 实例 化 UserModel 类 ， 那 
么 正确 的 实例 化 方式 如 以 下 代码 所 示 。 


public function index (){ 


Suser=D ("User"); 
Srows=$user->infoData(1); 
var_dump ($rows);// 输 出 调用 结 
} 
M 函数 同样 可 以 实例 化 目 定义 模型 ， 但 该 函数 必须 先 初 始 化 数据 库 ， 然 后 才能 实例 化 目 
定义 模型 类 ， 如 以 下 代码 所 示 。 
public function index (){ 
$user=M ("User","UserModel"); 


$rows=$user->infoData (1); 


var_dump ($rows); 


) 

M 函数 可 以 方便 地 进行 数据 库 操 作 。 利 用 目 定 义 模 型 能 够 实现 更 复杂 的 数据 库 操 作 。M 
因数 的 第 1 个 参数 表示 数据 表 名 ， 不 能 为 空 ;第 2 个 参数 为 日 定义 模型 类 ， 如 朵 载 入 日 定义 
模型 ， 该 模型 类 必须 位 于 当前 项 目 。 


O 提示 : ThinkPHP 框 染 使 用 的 是 单 入 口 文件 方式 ， 并 且 强 制 使 用 类 文件 目 动 载 入 机 制 ， 所 以 在 所 有 的 
MVC 开发 中 均 不 需要 开发 者 手动 3| 入 类 文件 。 


6.2.3 ”模型 初始 化 (_initialize()) 


在 实际 应 用 开发 中 经 党 需要 初始 化 一 些 全 局 接口 。 例 如 网 站 的 会 员 系 统 ， 在 访客 进入 会 
员 系 统 前 一 般 都 需要 判断 该 用 户 的 权限 、 级 别 ， 以 便于 系统 分 配 相应 的 功能 ， 再 比如 一 些 第 
三 方 扩 展 类 ， 在 调用 目 定 模型 之 前 都 需要 初始 化 。 在 标准 的 PHP 类 设计 中 ， 通 常 开 发 人 员 
会 在 构造 函数 中 完成 这 些 初始 化 的 操作 。 但 在 ThinkPHP 自 定义 模型 类 中 ， 并 不 允许 使 用 构 
造 函 数 ， 只 提供 了 _initialize0 方 法 ， 该 方法 拥有 最 高 的 优先 级 ， 所 有 的 目 定 义 类 只 要 存在 
_initialize0) 方 法 ， 就 会 首先 运行 该 方法 。 如 以 下 代码 所 示 。 

Public function _initialize(){ 


// 初始 化 的 时 候 检查 用 户 权限 
$this->checkRbac () ，; 
} 
_initialize0 不 仅 可 以 运行 在 自 定 义 模型 中 ， 还 可 以 在 控制 器 中 使 用 ， 得 到 的 效果 也 是 一 
样 的 。 利 用 _initialize(0) 方 法 可 以 方便 地 对 目 定 义 类 进行 初始 化 ，ThinkPHP 内 置 的 许多 扩展 也 
是 使 用 _initialize0 来 实现 的 ， 在 后 面 的 内 容 中 将 会 经 常 使 用 到 _initialize0) 方 法 。 
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6.3 ”控制 器 ( Controller ) 


控制 器 负 贡 MVC 的 任务 请 求 、 分 派 、 数 据 转换 等 操作 ， 最 后 将 结果 输出 给 视图 ， 完 成 
整个 MVC 的 请 求 流程 。 控 制 器 是 整个 MVC 处 理 流 程 中 最 重要 的 环境 ， 在 ThinkPHP 中 将 
控制 右 称 为 模块 ， 之 所 以 称 之 为 模块 是 因为 控制 占 在 URL 中 并 没有 具体 的 呈现 ， 用 户 看 到 
的 网 页 是 控制 器 里 面 的 动作 (Action)， 一 个 动作 通 第 对 应 一 个 网 页 ， 而 一 个 控制 右 可 以 包含 
名 和 干 个 动作 ， 事 实 上 无 论 称 为 模块 、 频 道 或 栏目 等 都 是 为 了 助 记 ， 只 要 该 类 遵循 文件 命名 规 
范 ， 并 继承 于 Action 类 ， 在 MVC 开发 中 都 是 指控 制 右 。 


6.3.1 创建 控制 器 


创建 一 个 控制 右 大 体 上 可 分 为 两 个 步 又， 分 别 为 控制 器 类 体 本 喘 以 及 控制 器 动作 。 所 谓 
的 控制 器 动作 就 是 类 体 中 的 方法 。 在 控制 嚣 中， 公开 为 publice 的 方法 都 可 作为 动作 ， 人 否则 只 
能 当成 成 员 方 法 使 用 。 控 制 器 文件 位 于 Lib/Action 目录 下 ， 一 个 最 简单 控制 器 只 需要 继承 于 
Action 基 类 即 可 ， 如 以 下 代码 所 示 。 


<?php 


class UserAction extends Action { 
public function index (){ 
/ DE ARIS 
} 


在 ThinkPHP 中 ， 控 制 器 类 名 都 是 带 Action 后 绥 修 饰 符 的 ， 控 制 器 名 称 一 定 程 度 上 决定 
T URL 的 访问 方式 。 创 建 控制 器 需要 遵循 的 原则 如 下 。 

> 默认 情况 下 ，ThinkPHP 对 控制 器 的 访问 是 区 分 大 小 写 的 ， 但 是 控制 器 的 类 名 首 字母 
必须 大 写 ， 为 了 使 URL 更 友好 ， 可 以 在 配置 项 中 将 URL_CASE_INSENSITIVE 设置 
为 true， 关 闭 大 小 写 检 测 。 

> 一 个 项 目 通常 需要 一 个 公共 的 控制 器 ， 以 利于 代码 重用 ， 如 果 所 有 的 控制 器 都 继承 
于 Action 基 类 ， 那 么 当 需 要 深入 改动 时 上 只 能 改动 Action 基 类 ， 这 是 不 现实 的 。 理 想 
的 做 法 是 在 一 个 项 目 中 创建 一 个 公共 控制 磊 (如 PublicAction )， 然 后 让 公共 控制 器 
类 继承 于 Action 基 类 ， 在 创建 自 定义 控制 费时 继承 于 公共 控制 费 类 ， 实 现代 人 码 的 高 
效 重用 ， 尤 其 在 一 些 需 要 做 权限 的 项 目 中 例如 会 员 系 统 、 后 台 管 理 等 )， 更 需要 有 
一 个 掌控 全 局 的 控制 器 。 


6.3.2 ”控制 器 中 的 动作 (Action) 


动作 是 控制 器 的 具体 行为 ， 一 个 没有 动作 的 控制 器 是 没 任何 意义 的 。 任 何 需要 输出 到 视 
图 中 的 数据 都 必须 由 动作 完成 。 默 认 情 况 下 ，ThinkPHP 会 生成 Index 控制 器 和 Index 动作 ， 
在 动作 中 只 需要 成 功 提示 信息 ， 如 以 下 代码 所 示 。 


<?php 
// 本 类 由 系统 目 动 生成 ， 仪 供 测 试用 途 


class IndexAction extends Action { 
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public function index (){ 
header ("Content-Type:text/html; charset=utf-8"); 
echo '<div style="font-weight:normal;color:blue;float:left;width:345px;text- 
align:center;border:lpx solid silver;background:#E8EFFF;padding:8px; font-size:14px; font- 
family:Tahoma">^_^ Hello, 欢迎 使 用 <span style="font-weight:bold;color:red">ThinkPHP</span> 
EGR 


} 


} 

默认 生成 的 index 动作 是 没有 任何 意义 的 。 使 用 过 Symfony 或 者 CakePHP SAER HITE 
者 ， 也 许 已 经 感觉 到 了 ThinkPHP 对 动作 处 理 的 灵活 性 。ThinkPHP 的 动作 和 一 个 普通 的 
PHP 类 方法 并 没有 多 大 区 别 ， 它 能 够 直接 输出 信息 ， 不 需要 借助 模板 ， 也 不 需要 分 配 变 量 ， 
开发 人 员 可 以 使 用 标准 的 PHP 代码 输出 HTML 等 信息 (Action 类 也 提供 了 show 方法 用 于 直 
接 输 出 )。 

尽管 ThinkPHP 人 允许 直接 在 动作 中 输出 信息 ， 但 这 并 不 是 一 个 好 的 编程 方式 。 动 作 的 设 
计 原 则 是 用 来 做 逻辑 运算 ， 所 以 应 尽量 避免 在 动作 中 混入 HTML 等 非 PHP RI, Action 基 
类 本 号 提供 了 display 方法 ， 用 于 结果 输出 。 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 


public function index (){ 
sthis->assign ("data", "欢迎 使 用 ThinkPHP")， 
$this->display (); 

} 


display 是 View 视图 引擎 中 一 个 好 用 和 禹 效 的 成 员 方 法 ， 在 Action 类 中 已经 对 display 
Um" TNS, (Efa display 能 够 智能 地 解释 当前 动作 视图 模板 ， 并 将 PHP 变量 、 数 组 等 信息 
输出 到 目标 文件 中 。display LA 3 种 局 效 的 模板 处 理 方式 ， 下 面 分 别 进行 介绍 。 

1. 调用 当前 控制 器 其 他 动作 模板 

默认 情况 下 ，display 会 查找 当 前 动作 对 应 的 模板 ， 当 然 也 可 以 手动 指定 需要 解释 的 模 
板 ， 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 


public function index (){ 
$sthis->assign ("data", "欢迎 使 用 ThinkPHP")， 
$this->display ("home"); 
} 
public function home (){ 
$this->display () 
} 


FJR EMERI R RRRA, A R VCF RNE O F A RRA 
件 都 是 以 .html 作为 后 级 )。 上 述 代 人 码 使 用 home 模板 ， 该 模板 必须 位 于 当前 控制 器 模板 目录 
D CDU TpVIndexhome.html)。 通 过 这 种 方式 可 以 实现 访问 不 同 动作 呈现 相同 模板 ， 但 不 同 结 
果 的 效果 ， 所 以 home.html 文件 也 是 可 选 的 。 

2. 调用 其 他 控制 器 动作 模板 

在 ThinkPHP 模板 方案 中 ， 一 个 控制 占 对 应 的 模板 方案 就 是 一 个 分 类 【〔( 即 一 个 文件 
夹 )， 由 于 跨 控 制 占 调用 就 意味 看 跨 文 件 夹 读 取 模板 。 所 以 在 使 用 display 解释 模板 时 ， 需 要 


E 国 127 


PHP MVC 开发 实战 


使 用 “: ”分 隔 符 ， 才 能 路 文件 到 解释 ， 如 以 下 代码 所 示 。 
public function index (){ 

sthis->assign ("data", "欢迎 使 用 ThinkPHP")， 

if (empty (cookie ("usernames")))It 
$this->display ("User:control"); 

}else{ 
Şthis->display ();}; 

} 


} 

这 里 所 指 的 其 他 控制 器 是 指 当前 项 目下 的 控制 器 ， 如 果 存 在 多 个 项 目 ， 虽 然 display 也 
能 够 文 持 跨 项 目 调用 ， 但 这 会 造成 项 目 结构 的 混乱 ， 降 低 代 码 可 读 性 ， 所 以 并 不 建议 跨 项 目 
调用 。 

3. 直接 输出 模板 文件 

display 还 能 够 文 持 直接 输出 一 个 静态 文件 ， 直 接 输 出 文件 不 受 ThinkPHP 内 置 的 模板 目 
录 处 理 方式 限制 ， 只 要 给 出 相应 的 模板 文件 路 径 ，display 就 能 够 将 该 文件 进行 解释 。 需 要 注 
意 的 是 ， 直 接 输出 模板 中 的 文件 必须 为 当前 网 站 内 的 文件 ， 而 不 能 是 远程 文件 ， 如 以 下 代码 
所 示 。 


public function head () { 
Şthis->display ("./Public/html/head.htm1"); 


} 

通过 前 面 的 内 容 介 绍 ， 可 以 看 到 display ICH RIMEN, RERI E L AN H ARI 
求 。Display 不 仅 能 够 支持 输出 HTML 网 页 ， 通 过 第 3 个 参数 还 能 够 输出 XML, WML 等 文 
件 ， 如 以 下 代码 所 示 。 


EE SE Et E VEfE=8!, Vexe,/ mL ) 


6.3.3 FEl 2S H9 H 


在 ThinkPHP 中 ， 控 制 器 是 允许 互相 调用 的 。 大 体 上 可 分 为 两 种 调用 方式 : 一 种 为 当前 
项 目 控 制 之 间 的 调用 ; 另 一 种 为 跨 项 目的 控制 器 调用 。 被 调用 的 控制 器 在 初始 化 后 ， 束 可 以 
当成 一 个 实例 类 来 使 用 。 为 了 便于 使 用 ， 系 统 还 提供 2 个 快捷 函数 处 理 控 制 器 的 调用 ， 下 面 
结合 示例 代码 分 别 介 绍 这 2 个 快捷 函数 的 使 用 方式 。 

1. A 函数 

A 函数 用 于 实例 化 控制 器 ， 它 的 作用 相当 于 new, A 函数 不 仅 支持 调用 本 项 目 内 的 控制 
器 ， 还 支持 跨 项 目 调 用 控制 器 ， 如 以 下 代码 所 示 。 


class IndexAction extends Action { 


public fumet ionm TeSt |) i 
Şobj=A ("Member"); 
Sobj->user () ; 

} 

} 


?> 


如 上 述 代 人 码 所 示 ，A("Member") 表 示 实 例 化 本 项 目 中 的 Member 控制 器 ;然后 调用 
Member 控制 器 中 的 user 方法 ， 该 方法 代码 如 下 所 示 。 

<?ph 

Deg MemberAction extends Action 


public function user () { 
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SthaLs=>assLcm DER e Be Ee 
Eh display) 
} 


PE BARA talk SE EER EKAN R, WREST EMRE, iHD ARRA 
test.html (ER TplIndex/test.html) 那么 系统 将 会 抛 出 错误 。 因 为 在 user 方法 中 使 用 了 $this- 
>display()， 这 了 束 意 味 看 在 当前 动作 (test 动作 ) 中 输出 模板 《〈 因 为 此 处 的 user 动作 已 被 实例 
化 ， 所 以 user 并 不 能 作为 动作 ， 而 是 作为 类 方法 )。 

如 果 和 需要 调用 的 控制 右 不 在 当前 项 目下 ， 那 么 需要 使 用 “://” 分 阳 和 人行。 此 外 ， 如 果 项 目 
使 用 模块 分 组 方式 ， 需 要 在 控制 右前 加 上 “7/” 分 隔 符 ， 路 项 目 控制 堪 的 调用 方式 如 以 下 代 
代 所 示 。 


class IndexAction extends Action { 


publie function test |) 4 
$obj=A ("Admin://Member"); 
Şobj->index (); 

} 

} 


2. R 函数 

A 函数 只 是 负责 调用 控制 器 (实例 化 类 )，R 函数 提供 更 加 简洁 的 操作 。R 函数 可 以 在 
调用 控制 器 时 指定 调用 方法 〈 动 作 )， 简 化 了 操作 步骤 。R 函数 的 使 用 格式 如 下 。 

R([ 项 目 名 :/][ 分 组 名 /模块 名 /操作 名 ',array( 参 数 1 ,参数 2…)) 

下 面 将 使 用 示例 人 代码， 演示 R 函数 的 使 用 。 


class IndexAction extends Action { 


publie EE cest |) 
R ("Member/user"); 
} 
} 


EK 


如 上 述 代 码 所 示 ， 表 示 实 例 化 当前 项 目 Member 控制 器 ， 然 后 调用 该 控制 器 下 user 动 
作 。 如 果 需 要 调用 路 项 目的 控制 器 ， 需 要 在 控制 融 前 加 上 “:/” 分 隅 符 ， 如 以 下 代码 所 示 。 


class IndexAction extends Action { 


publice rwnetion Test|) i 
R("Admin://Member/user"); 
} 
| 


?> 


上 述 代码 表示 实例 化 Admin 项 目下 的 Member 控制 器 ， 然 后 调用 user 动作 。 如 果 使 用 
模块 分 组 方式 部 普 应 用 ， 那 么 需要 在 控制 右前 加 上 “/” 分 隅 符 。 


6.3.4 项 目 空 控制 器 与 控制 器 空 动作 

在 一 些 网 站 中 ， 我 们 经 常 看 到 当 访 问 一 些 不 存在 或 者 过 期 的 网 页 时 ， 束 会 弹出 友好 的 
提示 说 明 。 这 是 利用 Web 服务 器 的 404 错误 处 理 机 制 实现 的 ， 所 有 主流 的 Web 服务 器 都 
提供 了 这 项 功能 。 在 ThinkPHP 中 ， 开 发 人 员 还 可 以 利用 空 控 制 器 与 空 动作 实现 错误 404 
的 功能 。 不 仅 如 些 ， 空 控制 器 与 空 动作 更 加 灵活 ， 因 为 它 也 是 一 个 真实 的 实例 类 和 方法， 
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能 够 实现 MVC 所 有 的 功能 ， 例 如 在 错误 出 现时 可 以 将 错误 信息 插入 数据 库 ， 提 供 个 性 化 
的 提示 等 。 

当然 ， 衬 控制 喜与 宇 动 作 是 不 能 代 符 服务 天 的 氏 误 页 面 处 理 机 制 的 。 至 控制 器 与 至 动作 
只 能 处 理 MVC 框架 内 的 页 面 ， 不 能 处 理 框 架 之 外 的 页 面 。 空 控制 右 与 空 动作 不 能 处 理 URL 
REWRITE 目 定 义 格式 的 页 面 。 下 面 在 将 分 别 介绍 空 控制 右 与 空 动作 。 

1. 空 控制 器 

当 用 户 所 访问 的 URL 不 存在 需要 访问 的 控制 项 时 ， 衬 控制 闫 融 派 上 用 场 了 。 假 设 用 户 
访问 的 URL 为 http:Wtp.localhosUindex.php/bbs， 但 该 项 目 内 并 不 存在 BbsAction Ehr, HB 
么 系统 将 会 给 出 错误 提示 ， 如 图 6-2 所 示 。 


系统 发 生 错误 


您 可 以 选择 [ 重 试 ] [返回 ] 或 者 [ 回 到 首页 ] 


错误 位 置 : FLE: D:\Downloads\Wxampp-win32-1.7.4-YC6\xampp\htdocs\tp3_demo\ThinkPHP 
\Lib\Core\App.class.php LNE: 159 


[ 错误 信息 ] 
无 法 加 载 模块 Bbs 
[TRACE ] 
[12-09-22 11:21:09] DADownloads\xampp-win32-1.7.4-YC6\xamppihtdocs\tp3_demo\ ThinkPHP 


\Lib\Core\App.class.php (159) Appzexec0 
[12-09-22 11:21:09] DADownloads\xampp-win32-1L7.4-YC6\xamppihtdocs\tp3_demo\ ThinkPHP 


图 6-2 ”控制 器 不 存在 提示 


需要 注意 的 是 ， 在 调试 时 务必 开局 调试 模式 〈 即 在 入 口 文件 中 定义 define("APP_ 
DEBUG",，true))， 人 奋 则 只 会 呈现 一 片 空 日 。 如 图 6-2 所 示 ， 信 息 提 示 Bbs RER Ghar) 并 
不 存在 ， 项 目 停 止 执行 。 接 下 来 在 项 目 Lib/Action 目录 下 创建 EmptyAction.class.php 控制 器 
类 文件 ， 如 以 下 代码 所 示 。 


<?php 
class EmptyAction extends Action{ 


public function index (){ 
ŝthis->assign ("msg", "你 所 查看 的 栏目 己 经 不 存在 "); 
E Dee SE lee ee E 
} 
// 罕 控制 右 类 方法 


EE EE Ee 


} 
E 
} 


再 次 访问 http://tp.localhost/index.php/bbs 网 址 ， 系 统 将 会 直接 调用 Empty 控制 器 。 开 
发 人 员 完 全 可 以 在 空 控制 器 中 定义 更 多 的 私有 成 员 方 法 ， 实 现 功 能 强大 的 自 定义 404 信息 
提示 
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2. Säit 

FAJE ( emm 1 与 衬 控 制 右 一 样 ， 都 是 系统 为 了 避免 URL 异 稼 而 设计 的 。 与 空 欣 制 
项 定位 到 栏目 《分 闫 ) 不 一 梓 ， 罕 动作 将 定位 到 其 体 的 页 和 面 ， 比 如 用 户 需 要 访问 
http://tp.localhost/index/user.htm (使 用 REWRITE URL 模式 ， 相 当 于 http://tp.localhost/ 
index.php/index/user)。 但 Index 控制 器 中 并 不 存在 user 动作 ， 那 么 只 需要 在 Index Joel ärt 
加 入 空 动作 ， 系 统 束 会 把 _emtpy IERE user 动作 。 如 以 下 代码 所 示 。 


<?php 
// 本 类 由 系统 目 动 生成 ， 仪 供 测 试用 途 
class IndexAction extends Action { 


public function _emtpy (){ 
sthis=>assiontnnsg"n 和 你 所 但 看 的 页 面 并 不 存在 四 7 
Şthis->display ("./Public/html/error.html1"); 


D 


6.3.5 ”动作 的 前 后 操作 


通 和 党 情况 下 ， 用 户 访 问 控制 器 动作 时 ， 系 统 会 直接 调用 动作 ， 然 后 由 display 方法 输出 
模板 。 这 个 过 程 是 顺序 执行 的 ， 开 发 人 员 需 要 在 执行 动作 之 前 或 者 执行 完 动 作 之 后 能 入 一 些 
额外 功能 代 但 ， 上 只 能 在 动作 中 ， 投 照 由 上 到 下 的 顺序 进行 添加 。ThinkPHP 对 控制 右 的 解释 
顺序 引入 了 动作 前 置 ( before ) 和 后 置 (_after_ ) 方法 ， 使 得 添加 前 置 代 码 和 后 置 代 人 码 弯 
mim. GI. 

采 置 方法 格式 如 下 。 

_before_ 动 作 名 

后 置 方法 格式 如 下 

_after_ 动 作 名 

下 面 通 过 示例 代码 ， 渤 示 动 作 前 置 方法 和 后 置 方法 的 使 用 。 


<?php 


class IndexAction extends Action { 
public function _before_index (){ 
echo "MPRE T, 
} 


publie function afcer imncex() 4 


See 
} 
public function index (){ 
$sthis->assign ("data", "欢迎 使 用 ThinkPHP")， 
Sen -display (I; 
} 


?> 
前 后 动作 的 顺序 与 运行 的 顺序 无 天。 如 上 述 代码 所 示 ， 系 统 在 调用 index 动作 之 前 会 首 
先 运 行 _before_index 方法 ; 然后 再 运行 index 动作 ;最 后 才 到 after index 方法 。 
需要 注意 的 是 ， 在 _before_index 前 置 方法 中 如 果 使 用 exit、$this->success 等 中 断 语句 ， 
那么 _after_index 后 置 方法 将 不 会 再 执行 ， 但 index 动作 不 受 影响 。 前 置 方 法 和 后 置 方 法 都 可 
以 使 用 $this->displayO 输 出 模板 ， 但 通 利 情况 下 前 置 和 后 置 方法 都 是 用 来 辅助 index 动作 运算 
的 ， 所 以 应 该 尽量 避免 得 出 html 等 代码 。 
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6.4 ”视图 (View ) 


AAA, EMRAT A REAR S o MVC 设计 蛋 式 中 最 核 
心 的 概念 就 是 视图 与 逻辑 分 离 ， 本 市 将 会 介绍 ThinkPHP 内 置 的 视图 引擎 ， 熟 悉 框架 的 视图 
处 理 方式 ， 将 能 够 有 效 提 高 谈 计 人 员 和 开发 人 员 的 工作 效率 。 


6.4.1 创建 和 使 用 视图 


视图 ， 通 俗 地 讲 束 是 网 页 。 视 图 的 呈现 需要 视图 引 敬 来 运算 ， 在 主流 的 PHP MVC 框架 
中 都 有 各 目的 视图 解释 引擎 ， 最 常见 的 是 使 用 标准 的 PHP 来 解释 ， 还 有 XML, XHML, 
Smarty、TagLib 等 。 对 于 Smarty， 相 信和 接触 过 PHP 的 读者 都 已 经 有 所 了 解 ， 这 里 需要 重点 
理解 TagLib 及 XML 解释 方式 。ThinkPHP 的 模板 引擎 高 效 之 处 在 于 灵活 的 视图 标签 ， 熟 悉 
这 些 标签 的 使 用 方式 将 能 够 提高 视图 模板 的 设计 水 平 。 接 下 来 将 从 基本 的 创建 视图 开始 ， 然 
后 深入 介绍 默认 视图 引擎 的 使 用 。 创 建 一 个 视图 可 分 为 以 下 几 个 步骤 。 

首先 确定 视图 模板 的 分 类 及 模板 的 存放 位 置 ， 例 如 Tpl/Index/index.html， 其 中 Index 对 
应 控制 占 ; index.html 对 应 控制 器 方法 。 如 果 使 用 模块 分 组 的 方式 ， 需 要 在 Index 目录 前 加 
上 对 应 的 分 组 名 称 目 录 。 系 统 通过 DEFAULT_THEME 配置 项 指定 项 目 视 图 模板 的 主题 ， 默 
认为 空 主题 。 应 用 多 主题 的 好 处 是 可 以 方便 地 在 网 站 中 应 用 多 种 模板 。 打 开 index.html， 代 
Din Fra, 


<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> 
<title> Sticlet</tircle> 

</head> 

<body> 

<h1> 这 是 一 个 测试 </h1> 

</body> 

</html> 


上 述 代 码 中 ， 在 视图 模板 中 使 用 了 变量 {$title}， 该 变量 的 值 由 控制 器 方法 提供 。 需 要 注 
意 的 是 ， 使 用 默认 的 {} 边 界 符 ， 在 视图 模板 中 使 用 JavaScript 等 脚本 时 ， 有 可 能 造成 冲突 。 
安全 起 见 ， 这 里 需要 将 默认 的 变量 符号 改 成 <!--{$}-->，Conf/config.php 配置 信息 如 下 。 


<?php 
return array 人 
'TITMPL_L_DELIM' => Yal- 
'TMPL_R_DELIM' GR GE 
); 
D> 


读者 可 以 根据 需要 进行 修改 ， 另 外 建议 将 边界 符 配 置 放 到 全 局 配置 文件 中 。 本 书后 面 的 
章节 内 容 中 均 使 用 该 配置 信息 ， 在 此 了 予以 说 明 。 此 时 index.html 模板 代码 如 下 所 示 。 

<html> 

<head> 

<rictle><l=-{ $title }==></trit le> 

</head> 
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<body> 

BEA Ua ane 

</body> 

</html> 

最 后 ， 需 要 在 控制 器 方法 中 使 用 assign 为 变量 赋值 ， 如 以 下 代码 所 示 。 

public function index (){ 

Sthis->assign(ntitlen "欢迎 进入 首页 ")， 

} 

assign 是 默认 视图 引擎 的 一 个 实例 方法 ， 该 方法 用 于 分 配 变 量 ， 变 量 的 值 可 以 是 PHP 文 
持 的 数组 、 对 象 或 者 PHP 变量 。ThinkPHP 内 置 了 多 种 视图 引擎 ， 在 默认 情况 下 系统 会 使 用 
内 置 的 视图 引擎 ， 如 有 果 使 用 第 三 方 的 视图 引擎 ， 变 量 的 分 配方 式 会 有 所 不 同 。 

视图 引擎 的 最 终 运行 结果 将 转换 成 标准 的 PHP，<! 一 {f}--> 边 界 符 也 会 被 解释 成 标准 的 
PHP 变量 ， 如 以 下 代码 所 未。 

<?php if (!defined('THINK_PATH')) exit ();?><html> 

<head> 


<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> 
<seictle><?phnpo echo EE ?></rirtle> 

</head> 

<body> 

<h1> 这 是 一 个 测试 </h1> 

</body> 

</html> 


ThinkPHP 扶 认 的 视图 引擎 支持 标准 的 PHP 代码 。 如 宋 使 用 PHP 代码 ， 模 板 标签 的 转换 
步骤 将 省 略 ， 从 而 在 一 定 程度 上 提高 性 能 。 要 启用 PHP 代码 解释 ， 需 要 修改 配置 项 
TMPL_DENY_PHP 值 为 false 或 者 TMPL_ENGINE_TYPE' =>PHP'。 


6.4.2 ”系统 变量 与 常量 


前 面 介绍 了 日 定义 变量 的 分 配 和 使 用 ，ThinkPHP HEERA G pE T IEY L AAE E 
量 ， 在 模板 中 使 用 这 些 内 置 变量 和 常量 ， 能 够 有 效 提 高 开发 效率 。 下 和 面 将 分 别 进行 介绍 。 

1. 系统 变量 

系统 变量 是 框架 内 置 的 变量 ， 并 已 经 被 赋值 。 开 发 人 员 在 模板 中 调用 时 只 能 得 看 或 使 用 
变量 值 ， 而 不 能 修改 变量 值 。 调 用 系统 变量 与 调用 目 定 义 变量 不 同 ， 调 用 系统 变量 需要 使 用 
$Think 关键 字 ， 如 <! 一 {$Think.get.username}--> 表 示 使 用 GET 变量 接收 URL 传 参 。 常 见 的 
ECKE CHE 6-1 所 示 。 


表 6-1 系统 内 置 的 变量 


变量 名 称 作 用 
$Think.get.name 相当 于 PHP 的 $_GET[‘name’] 
$Think.post.name 相当 于 PHP 的 $_POST[‘name’] 
$Think.server.name 相当 于 PHP 代码 的 $_SERVER[‘name’] 
$Think.session.name 获取 Session 值 ， 相 当 于 PHP 代码 的 Session::get(‘name’) 
$Think.cookie.name 获取 Cookie 值 ， 相 当 于 PHP 代码 的 Cookie::get(‘name’) 
$Think.config.name 获取 配置 文件 信息 ， 相 当 于 PHP 代码 的 CCmname”) 
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CÈ) 
变量 名 称 Ir D 
$Think.constname 获取 系统 常量 
$Think.lang.name 获取 语言 包 ， 相 当 于 PHP 代码 的 L(‘name’) 


为 了 便于 操作 ， 系 统 允许 开发 人 员 将 $Think KEF, W<!—{$Think.get.username }--> 
与 <! 一 {$get.username}--> 是 等 效 的 。 在 应 用 开发 中 读者 可 根据 实际 情况 灵活 使 用 这 些 内 置 的 
变量 。 

2. 系统 常量 

ThinkPHP 系统 还 内 置 了 许多 篆 量 ， 通 过 这 些 党 量 值 能 够 得 到 当前 项 目的 运行 信息 。 
ThinkPHP 视 几 引擎 允许 开发 人 员 在 模板 中 和 直接 从 入 一 部 分 音量 。 和 音量 由 关键 字 组 成 ， 它 不 
像 变 量 那 样 需 要 $ 符 写 ， 也 不 需要 标签 开始 从 和 结束 符 ， 下 接 在 模板 代码 中 输入 常量 名 称 即 
可 。 人 允许 在 模板 中 直接 使 用 的 音量 如 表 6-2 Bro, 


表 6-2 在 模板 中 使 用 的 常量 
用 


常量 名 称 作 
PUBLIC ` 获取 网 站 公共 目录 ， 系 统 默认 的 公共 目录 为 /Public 
_ TMPL ` 获取 当前 应 用 的 模板 目录 ， 默 认为 /项 目 /Tpl/ 

_ ROOT_ 获取 网 站 根 URL 

APP ` 获取 当前 项 目的 URL 

URL ` 获取 当前 栏目 〈 控 制 器 ) URL 
_ ACTION ` 获取 当前 动作 URL 

SELES 获取 当前 页 面 URL 


利用 第 量 能 够 获取 到 系统 环境 等 信息 ， 设 计 人 员 可 以 直接 在 模板 中 使 用 音量 代替 模板 中 
的 绝对 路 径 值 ， 让 模板 更 加 通用 。 如 以 下 代码 所 示 。 

<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 

<reiecle><l—=-{ $title }==></rit le> 

<link href=" _PUBLIC /default.css" rel="stylesheet" type="text/css" media="screen" /> 


<gcripr sre="<l1—-{ Şen ig- Url ==>/_ PUBLIC EE EEN 
</head> 
<body> 
</body> 
</html> 


上 述 代 人 码 还 利用 目 定 变量 <! 一 {$config.url}--> 获取 jquery.js 文件 ， 这 种 方式 能 够 确保 
无 论 模板 放 重 到 哪个 项 目 ， 或 者 变更 URL 和 个 式 ， 都 能 够 正确 地 获取 到 目标 资源 ， 最 终 的 表 
现形 式 为 http://tp.localhost/Public/js/jquery.js。 


6.4.3 ”在 视图 中 使 用 函数 


在 传统 的 PHP 编程 中 ， 开 发 人 员 可 以 方便 地 在 HTML 代码 中 骸 入 逻辑 代码 ，ThinkPHP 
模板 引擎 最 终 也 是 将 模板 中 的 特定 标签 转换 成 标准 的 PHP 人 代码， 这些 标签 可 以 是 语句 结 
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构 、 变 量 、 和 常量、 数组 等 ， 当 然 也 包括 函数 。 
假设 模板 中 有 一 个 变量 <! 一 {$addTime}-->， 该 变量 值 在 控制 器 动作 中 直接 赋 给 当前 时 间 
鹤 $this->assien(“addTime”,time())。 现 在 需要 输出 中 文 格式 的 日 期 及 时 间 。 如 果 使 用 传统 的 编 
EAA, MWARRBA kr, 
<?php 
// 使 用 PHP 编程 
Stime=date ("Y-m-d H:i:s",$addTime); 
echo $time; 


在 ThinkPHP RP AEH "IT ORSA RRG WREE =” Kee Dm ZS, 
上 述 代码 使 用 模板 标签 实现 ， 代 人 码 如 下 。 

<!--{$addTime |date="Y-m-d ER E 

如 上 述 代 人 码 所 示 ，$addTime 是 需要 输出 的 变量 ， 如 末 和 直接 输出 $addTime 将 会 得 到 一 串 
标准 时 间 惟 数字 。 由 于 使 用 “|” 引 入 了 date 函数 ， 该 函数 共有 2 个 参数 ， 每 个 参数 使 用 
“,” 隅 开 。 其 中 使 用 “ 捧 #” 符 号 表示 引用 变量 日 喘 值 。 最 后 得 出 的 结果 类 似 于 2012-09-24 
14:54:26 字符 串 。 

开发 人 员 还 可 以 引用 目 定 义 的 函数 ， 目 定义 函数 存放 在 项 目的 Common/common.php X 
件 中 ， 如 以 下 代码 所 示 。 

<?php 

// AE RŽI 

function formatTime ($time) { 

stime=date ("Y-m-d H:i:s",$addTime); 


return $time; 


} 


在 模板 中 调用 formatTime 目 定义 函数 和 调用 PHP 内 置 的 函数 是 一 样 的。 结合 后 面 介 绍 
的 多 语 襄 技术， 能够 轻松 地 实现 同一 模板 显示 多 种 时 间 格 式 的 功能 。 这 里 只 是 人 简 蛙 地 演示 在 
模板 中 怎样 使 用 函数 ， 在 实际 应 用 开发 中 ， 读 者 可 根据 需要 在 模板 标签 中 引入 函数 。 

在 模板 中 还 可 以 骨 侠 函数 ， 实 现 更 加 灵活 的 功能 。 函 数 租 僚 与 传统 的 PHP 一 样 能 够 
文 持 无 限 层次 。 在 ThinkPHP 模板 引擎 中 ， 使 用 函数 租 套 的 顺序 由 左 到 右 〈 传 统 的 PHP 
由 里 到 外 )， 即 模板 引擎 最 先 解释 左边 的 函数 ， 最 人 终 的 结果 由 最 右边 的 函数 决定 ， 如 以 下 
ARES PZR o 

as ele ee a 


上 述 代 码 转 换 后 的 PHP 代码 如 下 上 所 示 。 


EE EE E EE lee 


6.4.4 ”数据 循环 


利用 视图 引擎 的 assign 方法 可 以 将 包括 数组 在 内 的 数据 分 配 到 模板 处 理 。 在 模板 中 ， 
开发 人 员 可 以 使 用 传统 的 下 标 或 索引 取 到 数组 内 的 数据 ， 例 如 $list[“title”]。 这 种 方式 只 能 
获取 到 单一 的 数据 ， 如 末 一 个 数组 集合 中 存在 大 量 的 数据 (例如 数据 表 集 )， 那 么 就 需要 使 
用 循环 语句 逐条 谈 取 。ThinkPHP 默认 的 模板 引擎 提供 了 3 种 数据 循环 方式 ， 分 别 为 
volist, foreach 以 及 for 标签 。 这 3 对 标签 可 以 混合 使 用 ， 但 它们 都 有 各 上 自 的 适用 对 象 ， 下 
面 分 别 进行 介绍 。 
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1. volist 标签 

volist 标签 是 一 对 功能 强大 ， 使 用 方便 的 数据 循环 标签 ， 它 文 持 多 维 或 一 维 数 组 的 循 
环 。 由 于 它 容易 使 用 ， 并 且 语 法 灵活 ， 所 以 在 ThinkPHP 模板 中 该 标签 比较 常见 。volist 标签 
被 模板 引擎 解释 后 将 转换 成 foreach0...endforeach 语句 。 一 条 简单 的 volist 标签 语句 格式 如 
下 代码 所 示 。 


<volist name=" 数 据 源 " id=" 临 时 变量 名 "> 
// 循 环 输出 


</volist> 


如 上 述 代 人 码 所 示 ， 数 据 源 文 持 PHP 所 有 数组 (不 文 持 对 和 象 )， 临 时 变量 名 称 可 以 目 害 
义 ， 该 变量 名 决定 了 在 循环 体 中 可 以 使 用 的 数组 名 称 。 为 了 方便 演示 ， 这 里 将 在 控制 如 
index 动作 中 定义 一 个 多 维 数 组 ， 如 以 下 代码 所 示 。 


public function index (){ 


Şrows=array ( 
0=>array ( 
"user_id"=>1, 
"user_name"=>"xiaoming", 
"user_mail"=>"kfQ86055.com" 
ES: 
1=>array ( 
"user id"=>2, 
"user_name"=>"]ishi", 
"user mail"=>"ceiba @126.com" 
JS 
2=>array ( 
"user id"=>3, 
"user_ name"=>"wangwu", 
"user_mail"=>"mmg7Q@qq.com" 
Dy 
3=>array ( 
"user id"=>4, 
"user name"=>"zaoliu", 


"Ser Mall"=>"" 


); 
SEnls—>asSsSlenm (Vlist; Srows) 
Şthis=>cisplay () ; 


} 
上 述 代码 中 定义 了 1 个 数组 ， 并 定义 了 3 条 数据 。 在 index.htm 中 可 以 使 用 volist 标签 
将 效 组 数据 循环 输出 ， 如 以 下 代码 所 示 。 


<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
SEET Eeer 

</head> 

<body> 


<div class="main"> 
<volist name="list" id="vo"> 
<li><!-—-{Şvo.user_name}--></li> 
</volist> 


</div> 
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</body> 
</html> 


如 上 述 代码 所 示 ，list 为 数组 变量 ， 该 变量 由 index 动作 分 配 。 在 标签 中 直接 引用 变量 只 
需要 变量 名 即 可 ， 不 需要 定 界 符 和 “$?” 符 号 。$vo 是 一 个 临时 变量 (由 volist 标签 id 值 指 
定 )， 所 有 等 竺 循环 的 数据 都 被 存放 在 该 变量 中 ，$vo.user_ name 表示 输出 数组 中 的 
user_name 键 对 应 值 ， 读 者 还 可 以 像 传 统 的 PHP 那样 写成 $vo['user_name’]。 

在 实际 应 用 开发 中 ， 一 般 只 需要 使 用 volist 标签 的 name 和 id 属性 即 可 完成 大 多 数 的 数 
据 循环 。ThinkPHP 为 了 能 够 使 volist 标签 更 加 灵活 ， 还 提供 了 另外 一 些 实用 的 可 选 属 性 ， 这 
些 属性 如 下 。 

> offset: 起 始 数据 序列 。 
length: 数据 结束 序列 。 
key: 数据 循环 时 的 临时 变量 ， 相 当 于 从 0 开始 的 索引 号， 默认 变量 名 为 i。 
mod: 对 key 值 取 模 。 
empty: 当 name 数组 为 空 时 显示 的 字符 串 信 息 。 

offset 和 length 配合 使 用 能 够 实现 简单 的 数据 分 页 功能 ， 如 果 数 组 信息 过 多 ， 可 以 使 用 
offset 和 length 分 请 输出 ， 如 以 下 代码 所 示 。 


<volist name="list" id="vo" offse="2" length="6"> 
<li><!-—-{$$vo.user_name}--></li> 
</volist> 


上 述 代码 表示 只 循环 索引 吕 2 一 6 之 间 的 数据 ， 利 用 offset 和 length 属性 ， 可 以 方便 地 
控制 页 面 中 的 局 部 数据 。 

2. foreach 标签 

foreach 标签 是 volist 标签 的 简化 版 ， 使 用 方式 一 样 ， 最 终生 成 的 PHP 代码 也 一 样 。 唯 
一 不 同 的 就 是 foreach 标签 不 文 持 volist 中 可 选 属性 《只 文 持 key)。 男 外 ，foreach 不 仅 可 以 
循环 PHP 数组 ， 还 可 以 循环 对 象 。foreach 的 临时 变量 使 用 item (RÆ volist 中 的 id。 如 以 下 
代码 所 示 。 

<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 

ee el 


</head> 
<body> 


VW VY 


<div class="main"> 
<foreach name="list" item="vo"> 
<li><!-—-{$$vo.user_name}--></li> 
</foreach> 
</div> 
</body> 
</html> 
3. for 标签 
for 标签 也 是 一 种 可 以 实现 数据 循环 的 标签 ， 最 终生 成 的 代码 就 是 PHP 中 的 for 语句 
块 。 无 论 是 简洁 性 、 易 用 性 还 是 程序 的 运行 速度 都 不 及 volist、foreach 标签 。 但 for 标签 不 
仅 可 以 循环 数组 ， 还 可 以 循环 普通 的 PHP 和 变量、 函数 等 ， 第 用 在 局 部 统计 、 更 狐 等 场合 。 
for 标签 格式 如 下 。 
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<for start=" F iR" end=" 结 束 值 ” comparison="" step=" 步 进 值 " name=" 循 环 变 量 名 " > 
/ /数据 循环 


> 
一 条 最 简单 的 for 标签 语句 只 需要 开始 值 和 结束 值 即 可 ， 这 些 变量 值 都 支持 模板 或 者 控 
制 右 动作 中 的 变量 值 ， 如 以 下 代码 所 示 。 


Selen Deele 


<for start="0" end="$counts"> 
<li><!-—-{Ş$vo.user_name}--></li> 
</for> 
</div> 


在 for 标签 中 使 用 变量 与 volist、foreach 标签 不 同 ， 在 for 标签 中 使 用 变量 必须 在 变量 名 
REME “$”, THE "IT mër Hierk, for 标签 可 选 属 性 如 下 。 
> comparison: start 与 end 的 条 件 ， 默 认为 lt 小于)。 常 用 的 有 et (等 于 或 小 于 )、eq 
CT e 
> step: 步 进 条 数 。 
> name: 循环 时 的 变量 名 ， 默 认为 i。 


6.4.5 条件 判断 


虽然 多 数 的 运算 过 辑 都 在 控制 器 和 模型 中 完成 ， 但 在 模板 引擎 深 染 过 程 合理 地 进行 一 些 
逻辑 判断 不 仅 能 提 蜗 代码 的 可 读 性 ， 还 能 够 提 融 程序 的 运行 效 京 。 主 流 的 框架 大 多 数 都 支 持 
在 视图 模板 中 进行 逻辑 判断 、 比 较 等 基本 操作 ，ThinkPHP 默认 的 视图 引擎 提供 了 非常 丰富 
的 逻辑 运算 判断 、 比 较 等 标签 。 这 些 标签 灵活 易 用 ， 熟 悉 运算 标签 的 使 用 将 极 大 地 提供 开发 
效率 ， 同 时 也 让 后 人 台 代 码 更 加 规范 和 整洁 ， 下 面 分 别 介绍 。 

1. 运算 判断 标签 

在 传统 的 PHP 开发 中 ， 开 发 人 员 可 以 使 用 if 等 标签 比较 两 个 数值 的 大 小 ， 以 便 程 序 进 
行 下 一 步 的 运算 。 在 ThinkPHP 视图 引擎 中 ， 开 发 人 员 可 以 使 用 内 置 的 标签 实现 运算 判断 ， 
如 表 6-3 所 示 。 


表 6-3 判断 比较 标签 


标 ZS 合 ” ,以 PHP 表示 法 

eq CSR == 

neq A !=、<> 

St SS > 

egt 大 于 或 等 于 >= 

lt 小 于 < 

elt 小 于 或 等 于 <= 

heq a === 

nheq PEST !== 


IT D Smarty 模板 引擎 的 谈 者 相信 对 这 些 标签 并 不 阳 生 ， 这 些 标签 都 是 由 相应 的 单词 缩 
写 而 成 ， 如 neg 就 是 not equal 的 缩写 。 相 比 Smarty， 在 ThinkPHP 视图 模板 中 使 用 这 些 运 算 
比较 标签 更 加 方便 和 灵活 ， 如 以 下 代码 所 示 。 
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<volist name="1ist" id="vo"> 
<div class="main"> 
<egt name="vo.user_id" value="2"> 
<li><!-—-{$$vo.user_name}--></li> 
</egt> 
</div> 
</volist> 


如 上 述 代码 所 示 ， 在 volist 循环 体 中 使 用 egt 判断 标签 ， 表 示 只 要 输出 userid 大 于 或 等 
于 2 的 记录 。 其 中 name 属性 表示 比较 源 ，value 表示 比较 的 目标 ， 这 两 个 属性 都 支持 使 用 变 
量 (包括 系统 变量 )。 需 要 注意 的 是 ， 属 性 nme 在 设置 变量 时 不 需要 指定 “$” 符 号 ， 而 
value 属性 必须 指定 “$” 符 号。 判断 标签 最 终 将 由 模板 引擎 转换 成 PHP 判断 语句 ， 如 以 下 代 


但 所 不。 
<?php 
lene es A A EE 
| 


<?php endif; ?> 


其 他 的 判断 标签 使 用 方式 一 样 ， 在 此 不 做 细 述 。 判 断 标签 还 可 以 结合 <else/> 标 签 使 用 ， 
这 样 就 可 以 实现 类 似 于 if...else 的 效果 了 了， 如 以 下 代码 所 示 。 


SE hun se o> 


<div class="main"> 
<neq name="vo.user mail" value=""> 
lee user name =-></1i> 
<else/> 
<p1> 邮 箱 为 空 </p1> 
</neq> 
</div> 
</volist> 
2. if 5 switch 标签 
在 模板 设计 中 还 可 以 使 用 功能 更 加 丰富 的 让 be, if 标签 能 够 实现 运算 判断 标签 的 所 有 
功能 ， 并 且 人 允许 设计 人 员 在 比较 属性 里 加 入 PHP 代码 ， 也 可 以 在 比较 属性 里 使 用 现 有 的 功 
能 函数 。if 标签 虽然 功能 丰富 ， 但 使 用 起 来 相对 也 比较 烦琐 ， 下 面 将 通过 代码 演示 if 标签 的 
使 用 ， 如 下 代码 所 示 。 
<if condition=" (S$Sweek gt 0) and (S$Sweek elt 7) "> 
<if condition="$week eq 1"> 
星期 一 


EE 


<if Conditions Sweek eq 2"> 
星期 二 

SE 

<if Conditions Sweek eq 3"> 
星期 三 

SEET 

<if Conditions Sweek eq 4"> 
星期 四 

</if> 

<if condition="$week eq 5"> 
星期 五 

SE 

<if Conditions Sweek eq 6"> 
星期 六 
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Lf LES 
<else /> 
SREWR, WAH- 
SE EE 
上 述 代码 将 根据 控制 器 动作 分 配 的 week Jeep, Amt e HD IL. PhS 
算 代 码 如 下 所 示 。 
public function index (){ 
weekday = array COT, Ee 
Şweek =Ş$weekday[date('w', 5 _SERVER['REQUEST_TIME'])];}; 
Şthis->assign("week", Sweek 1: 
Şthis->display ();}; 
} 
与 if 标签 类 似 的 还 有 switch 标签， 使 用 switch PEKKI ERDA 
化 ， 如 以 下 代码 所 示 。 


<switch name="week" > 


G D 
GE 
DN 
ka 
KE 
H: 
JÀ 
也 
z3 


<case value="1"> 
星期 一 
</case> 
和 GaSS values" iz 
星期 二 
</case> 
<case value="3"> 
年 中 = 
</case> 
<case value="4"> 
星期 四 
</case> 
<case value="5"> 
星期 五 
</case> 
<case value="6"> 
星期 六 
</case> 
<default /> 
SREWA, WAH 
</switch> 
switch PRR J if br, EIRE DL PH OMRE 让 标签。 这 里 需要 注意 的 是 switch 
标签 中 的 name 属性 ， 该 属性 指定 数据 的 比较 来 源 ， 支 持 PHP 所 有 变量 (包括 系统 变量 )， 
但 不 需要 使 用 “$” 变 量 分 配 符号 ; 另外 ， 在 value 条 件 配对 中 ，switch 标签 允许 同时 对 多 个 
结果 进行 配对 ， 多 个 结果 之 间 使 用 “|” 分 隔 ， 如 value=”5|6”。 
if 标签 中 的 condition 属性 允许 直接 能 入 PHP 代码 ， 例 如 直接 使 用 函数 <if 
condition="$passwd eq md5($_post['password' ">, BJ; if 标签 功能 强大 ， 但 由 于 实现 比较 烦 
琐 且 容易 出 错 ， 所 以 建议 谈 者 尽量 使 用 其 他 标签 代 符 。 
本 方 重点 介绍 了 ThinkPHP 提供 的 利用 的 判断 标签 ， 系 统 还 有 一 些 实用 的 标签 ， 由 于 篇 
幅 所 限 ， 这 里 就 不 一 一 讲述 。 读 者 在 实际 应 用 开发 中 借鉴 本 节 内 容 再 配合 官方 提供 的 开发 手 
册 ， 相 信和 其 他 判断 标签 也 能 轻松 掌握 。 
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6.4.6 ”使 用 外 部 文件 


在 ThinkPHP F, EH include 标签 包含 页 面 。 与 PHP 的 include 不 同 ， 在 模板 中 使 
用 include 对 引擎 的 解释 过 程 不 受 任 何 影 响 ， 残 算 被 包含 的 文件 不 存在 也 不 会 被 中 断 执 行 。 
合理 地 使 用 include 可 以 让 页 和 面 变 得 整洁 ， 整 站 的 模板 维护 更 加 容易 ， 但 小 用 include 标签 也 
会 造成 代码 灾难 ， 降 低 模 板 的 通用 性 。ThinkPHP 还 提供 了 import 标签 及 load 标签 ， 这 些 标 
俭 作 用 类 似 ， 但 也 有 各 目的 适用 对 象 ， 下 面 分 别 介绍 。 

1. include 标签 

这 里 的 include 标签 与 PHP 的 include 语句 不 同 ， 模 板 中 的 include 模板 只 能 解释 静态 的 
hm 等 文件 ， 而 不 能 直接 包含 PHP 文件 ， 使 用 过 shtml MEE A AA AA <!--#include 
file=""top.html""--> 语 句 ， 该 语句 通常 用 于 包含 第 三 方 文件 ， 例 如 广告 代码 、 页 面 导 航 、 版 权 
说 明 等 。 这 里 的 include 标签 和 shtml 中 的 include 语句 作用 是 一 样 的 ， 只 不 过 形式 不 一 样 而 
已 。 一 条 最 简单 的 include 标签 语句 如 以 下 代码 所 示 。 

<include file="./Tpl/default/Public/header.html" /> 

这 里 需要 注意 的 是 ，include 包含 的 静态 文件 必须 位 于 当前 网 站 下 ， 不 能 是 一 个 URL, 
该 文件 是 一 个 完整 的 文件 路 径 ， 通 第 从 入 口 文件 算 起 。 另 外 URL 模式 尽量 不 要 使 用 日 定义 
的 URL REWRITE 模式 ， 人 否则 目录 层级 有 可 能 发 生 改 变 ， 寻 致 mclude 找 不 到 文件 。 

include 标签 不 仅 可 以 包含 完整 路 径 的 静态 文件 ， 也 可 以 包含 控制 器 动作 。 前 面 已 经 介绍 
过 ， 一 个 控制 右 动 作 束 相当 于 一 个 页 面 ， 所 以 使 用 include 标签 是 允许 直接 包含 控制 器 动作 
的 《动作 必须 使 用 this->display0 输 出 模板 )。 如 以 下 代码 所 示 。 

<include file="head" /> 

以 上 代码 表示 包含 当前 控制 若 的 head JE, WR head 动作 不 在 当前 控制 项 ， 那 么 需要 
使 用 “:” 分 隔 符 ， 如 以 下 代码 所 示 。 

<include file="News:head" /> 

在 包含 文件 的 同时 ，include 标签 还 支持 问 目 标 页 面 传递 额外 的 变量 参数 ， 这 些 参 数 会 被 
模板 引擎 解释 成 PHP 变量 ， 如 以 下 代码 所 示 。 

ET 

将 要 传递 的 参数 附加 到 file 属性 值 即 可 ， 这 些 参数 可 以 在 目标 页 面 head.html 文件 中 使 
用 “[]” 罕 写 进行 配对 ， 如 以 下 代码 所 示 。 

<html xmlns="http://www.w3.o0rg/1999/xhtml"> 

<head> 

<title>[title]</title> 


<meta name="keywords" content=" [keywords]" /> 
</head> 


利用 这 些 特 性 ， 设 计 人 员 可 以 方便 地 在 页 面 中 嵌入 外 部 网 页 ， 实 现 页 面 布 局 、 广 告 系统 
等 实用 的 功能 。 使 用 include 标签 关键 是 网 站 的 URL 模式 ， 读 者 需要 计算 好 路 径 ， 过 多 地 包 
含 控制 吉 动 作 会 导致 代码 维护 困难 ;在 部 闭 时 ， 如 果农 包含 的 页 面 发生 了 改变 ， 那 么 需要 开 
发 者 手动 删除 当前 视图 模板 缓存 ， 人 否则 将 看 不 到 最 终 效 末 。 

2. import 标签 

import 标签 与 include 标签 一 样 都 可 以 引用 外 部 文件 ， 但 import 标签 具有 和 针对 性 ，import 
标签 最 常用 于 导入 JavaScript 或 CSS 文件 ， 通 过 属性 type 指定 文件 的 类 型 就 可 正确 地 导入 文 
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件 ， 避 免 了 手动 编写 脚本 的 步 又， 提高 了 界面 的 整洁 性 。 如 以 下 代码 所 示 。 
<import type='js' file="Js.Util.Array" /> 
<import file="Js.Util.Array,Js.Util.Date" /> 
<import type='css' file="Css.common" /> 


import REFERIT, EH E” RRE A, 默认 的 路 径 由 Public 目录 开始 算 起 ， 文 
件 不 需要 后 缀 名 。 如 上 上面 的 Ji.UtlArray， 解 释 后 的 路 径 为 /Public/JSUUHlArray.js。import 文 
持 一 次 性 导入 多 个 文件 ， 多 文件 之 间 使 用 “,” 分 隔 符 。 

import 标签 通过 属性 type 指定 需要 导入 的 文件 类 型 ， 默 认 导 入 JavaScript 脚本 文件 ， 更 
改 type 属性 值 可 以 改变 导入 的 文件 类 型 ， 如 type=”css”。 

import 标签 以 包 的 形式 导入 网 站 中 公共 静态 资源 文件 ， 如 果 需 要 导入 第 三 方 网 站 或 者 使 
用 显 式 的 路 径 资源 文件 ， 可 以 使 用 load 标签 ， 该 标签 的 使 用 方式 和 import 标签 一 样 ， 如 以 
下 代码 所 示 。 


<load href="../Public/Js/Common.js" /> 
<load href="http://code.jquery.com/jquery-1.8.2.min.js" /> 


虽然 默认 的 模板 引擎 握 供 了 灵活 方便 的 文件 导入 标签 ， 但 如 前 面 所 述 一 样 ， 这 些 标签 最 
终 都 是 转换 成 标准 的 HTML 代码 ， 使 用 导入 标签 虽然 一 定 程 度 上 所 高 了 开发 效率 ， 但 页 面 
中 太 多 的 标签 一 起 使 用 也 需要 牺牲 运行 效率 。 


6.4.7 FAMAE 


前 面 介绍 的 标签 都 是 系统 内 置 的 标签 ，ThinkPHP 模板 引擎 标签 处 理 机 制 借鉴 了 JSPTag 
设计 思想 ， 提 供 了 灵活 的 标签 扩展 驱动 。 使 用 标签 扩展 ， 开 发 人 员 就 可 以 将 一 些 常 用 的 功能 
代 人 码 封 疙 为 标签 ， 在 使 用 时 束 像 使 用 内 置 标签 一 样 ， 和 直接 输入 标签 参数 即 可 完成 标签 的 调 
用 。 被 封装 为 标签 的 代码 可 以 是 PHP, JavaScript, HTML 等 ， 最 终 这 些 代码 将 被 模板 引擎 
转换 为 模板 中 可 用 的 脚本 代码 。 下 面 首 先 介绍 怎样 创建 目 定 义 扩展 标签 ， 然 后 再 讲解 怎样 在 
模板 中 使 用 目 定 义 标 签 。 

1. 创建 自 定 义 扩 展 标签 

存放 扩展 标签 的 目录 位 于 ThinkPHP/Extend/Driver/TagLib， 要 创建 日 定 扩 展 标 签 需要 在 
该 日 录 下 创建 相应 的 类 文件 。 和 其 他 常 见 类 库 一 样 ， 一 个 文件 类 代表 一 个 标签 类 库 ， 它 的 命 
名 方式 为 “TagLib+ 标 签 库 名 称 ”， 文件 后 级 名 为 .class.php。 为 了 方便 演示 ， 这 里 将 创建 一 个 
名 为 My 的 标签 类 库 ， 类 库 文 件 名 为 TagLibMy.class.php。 打 开 TagLibMy.class.php 文件 ， 首 
先 定义 类 库 类 名 ， 并 让 其 继承 于 TagLib 基 类 ， 如 以 下 代码 所 示 。 

<?php 


class TagLibMy extends TagLib{ 
} 


?> 

这 样 一 个 名 为 My 的 标签 类 库 束 可 以 被 系统 识别 了 。 只 有 类 库 没 有 标签 是 没有 和 意义 
的 ， 所 以 还 需要 为 该 类 库 制作 标签 。TagLib 类 提供 了 parseXmlAttr 扩展 接口 ， 该 方法 能 够 
将 数组 信息 解释 为 标签 所 需要 的 XML 属性 。 为 了 方便 讲解 ， 这 里 将 制作 一 个 人 简单 的 logo 
标签 ， 该 标签 用 于 显示 网 站 的 logo, HA 2 个 参数 ， 分 别 为 标签 的 id 以 及 logo 的 大 小 类 
型 ， 步 骤 如 下 。 


首先 在 tags 属性 中 定义 标签 属性 〈 即 参数 )， 这 里 共 定 义 2 个 属性 ， 如 以 下 代码 所 示 。 
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<?php 
class TagLibMy extends TagLib{ 
protected Stads = array 人 
eize H Logo Kb 
Ieren EE E Ee ee e "Closs =0) ; 
IR 


R 


如 上 述 代码 所 示 ， 在 tags 类 成 员 属 性 中 ， 定 义 了 一 个 名 为 logo 的 数组 ， 访 数组 索引 键 
束 是 标签 名 称 。logo 数组 的 值 为 关联 数组 ， 关 联 数组 内 的 数组 元 素 决 定 了 标签 的 功能 ， 其 中 
attr 指定 了 标签 的 属性 (参数 )， 其 他 的 数组 元 素 作 用 如 下 。 

> close: 标签 是 否 为 团 合 方 式 (0 闭合 1 不 闭合 )， 默 认为 不 闭合 。 

> leve: 标签 的 散人 套 层 次 (只 有 不 闭合 的 标签 才 有 骸 僚 层次 )。 

> alias: 标签 别名 。 

attr 数组 决定 了 目 定 义 标签 能 够 接收 的 属性 ， 如 果 和 需要 多 个 标签 属性 ， 只 需要 在 attr 数 
组 项 值 中 定义 多 个 数组 元 素 即 可 。 

定义 完 标 签 数据 后 ， 接 下 来 还 需要 定义 标签 的 处 理 方法 。 标 签 处 理 方法 就 是 一 个 普通 
的 实例 类 方法 ， 在 该 方法 内 需要 使 用 parseXmlAttr 将 数组 解释 成 XML 数据 ， 如 以 下 代码 
所 示 。 


public function _logo($attr, $content) 
{ 


Stad = $this->parseXmlAttr ($attr, 'logo'); 
Sid = lemocy (Stragi iatl) 2 SEagol Ge 3 logo’ 7 
ŠŚsize=$tag[|"size"]; 
switch (Ş$size){ 
eane Hopie Ne 
$str='<img name="logo" src="./Public/images/logo_big.jpg" width="200" 
height="140" alt=" 网 站 大 Logo"™>'， 
break; 
Case "ema": 
$str='<img name="logo" src="./Public/images/logo_small.jpg" width="60" 
height="35" alt=" 网 站 小 Logo">'， 
break; 
default: 
SSTr= "SLZ 参数 有 :big、 Snel ls 
} 
return S$str; 


} 

如 上 述 代 人 码 所 示 ， 标 签 的 处 理 方 法 命名 规则 为 “ 标签 名 ”， 这 是 系统 的 规范 。 上 述 代码 
的 功能 很 简单 ， 就 是 根据 logo 标签 中 的 size 属性 值 ， 返 回 logo 图 标 。 通 过 上 述 步骤 ， 一 个 
II DD GV alte TI, HF CR EH o 

2. 使 用 自 定 义 扩 展 标签 

使 用 目 定 义 标 签 和 使 用 系统 内 置 的 标签 ， 最 大 不 同 之 处 在 于 使 用 目 定 义 标 签 需 要 引入 标 
签 类 库 ， 并 且 需 要 声明 标签 前 缀 为 类 库 名 。 引 入 上 自 定 义 标签 类 库 需 要 使 用 Taglib 系统 标签 ， 
例如 引入 前 面 创建 的 My 类 库 ， 代 人 码 如 下 所 示 。 

<taglib name='My' /> 

引入 扩展 类 库 后 ， 了 驶 可 以 直接 使 用 了 ， 和 格式 为 “< 类 库 名 :标签 名 属性 />” 代 码 如 下 
所 示 。 
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<my: Logo Jeff lLndesz em Se 


通过 前 面 内 容 的 介绍 ， 相 应 读者 已 经 对 扩展 标签 有 了 一 个 清晰 的 认识 ， 也 体验 到 上 日 定义 

扩展 标签 的 灵活 P ThinkPHP 官方 开发 包 中 ， 提 供 了 众多 第 三 方 扩 展 包 ， 这 些 扩展 
包 多 数 都 是 利用 目 定 义 扩 展 标 签 来 实现 的 ， 例 如 系统 提供 的 证 文本 编辑 器 。 

在 模板 中 使 用 富 文 本 框 是 一 件 轻松 的 事 ，ThinkPHP 框架 本 身 就 提供 了 editor 标签 ， 该 
标签 能 够 直接 在 页 面 中 舱 入 编辑 框 。ThinkPHP 官方 的 editor 标签 位 于 tp 扩展 标签 库 中 ， 
在 使 用 前 需要 在 解压 包 中 找到 Examples/Tag/Lib/TagLib/TagLibTp.class.php 文件 ， 然 后 将 其 
复制 到 ThinkPHP/Extend/Driver/TagLib 目录 ， 最 后 在 模板 中 直接 骨 入 标签 即 可 ， 如 以 下 代 


公所 示 。 
<taglib name="tp,My" /> 
<html> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
ZE TE eS /Eel 
</head> 
<body> 


<tp:editor id="textContent" uploadurl="/Public/editor up" width="600"></tp:editor> 
<form id="form1" name="forml" method="post" action=""> 

<textarea name="content" id="textContent" cols="45" rows="5"></textarea> 

</form> 

</body> 

</html> 


EE ER 
tas P E ERRET E; width 属性 指定 编辑 器 的 显示 宽度 。 最 终 效果 如 图 6-3 所 示 。 


Lemma, 
Firefox ~ 
< 


Ke BIG BI JS e kd 
IL FLIT MIR U AD 
: | Te Tr 


HIM 
£ 


图 6-3 editor E zy A ar H 
利用 同样 的 原理 ， 读 者 还 可 以 能 入 其 他 好 用 的 编辑 器 ， 本 书 17.4.1 "0 atb OG vV br 
签 实现 百度 编辑 髓 先入 的 。 总 之 ， 系 统 提 供 的 标签 扩展 有 效 地 增强 了 模板 引擎 功能 ， 其 作用 
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类 似 于 Smarty 扩展 插件 ， 无 论 作 为 界面 设计 人 员 ， 还 是 后 合 PHP 开发 人 员 ， 痢 应 掌握 。 


6.4.8 ”使 用 布局 


在 主流 的 一 些 MVC 框架 中 都 提供 有 网 站 布局 功能 ， 如 前 面 介绍 过 的 Symfony、 
CodeIgniter、CakePHP 等 ， 这 些 框架 默认 的 模板 方案 就 已 经 启用 了 布局 功能 。 使 用 布局 能 够 
有 效 地 提高 模板 的 开发 速度 ， 并 且 由 于 布局 的 限制 ， 网 站 中 的 各 页 面 能 够 保持 高 效 的 统一 ， 
方便 后 期 维护 。 一 个 经 典 的 网 站 布局 如 图 6-4 所 示 。 

如 图 6-4 所 示 ， 这 是 一 个 经 典 的 网 站 布局 图 ， 如 果 网 站 
页 面 内 容量 大 ， 栏 目 多 ， 那 么 布局 束 会 更 加 复 淋 。 如 末 不 使 
用 布局 功能 ， 那 么 要 你 持 整 个 网 站 相同 的 布局 样式 ， 束 必须 
在 每 个 模板 中 使 用 相同 的 代码 ， 这 无 疑 会 极 大 地 增加 开发 成 
本 ， 降 低 开 发 效率 。 而 一 旦 使 用 布局 功能 ， 那 么 上 只 需要 将 
header, side, footer 区 域 的 重复 代码 放 到 布局 模板 中 ， 留 下 
content 区 域 的 代码 放 到 各 目 探 制 器 动作 模板 中 ， 那 么 就 能 够 图 6-4 网 站 布局 
EES ENEE 

ThinkPHP 3.0 之 前 的 模板 引擎 没有 布局 的 概念 ， 但 开发 人 员 可 以 使 用 include 包含 文件 
达到 网 站 布局 的 效果 。 新 版 的 ThinkPHP 引入 了 模板 布局 功能 ， 要 使 用 模板 布局 需要 在 配置 
项 中 打开 LAYOUT_ON 选项 〈 默 认为 关闭 ， 即 false)。 为 了 方便 演示 ， 接 下 来 将 使 用 一 个 简 
单 的 模板 文件 ， 泪 示 布 局 功能 的 使 用 ， 帮 助 读 者 加 深 对 模板 布局 的 认识 。 

首先 在 项 目 Tpl 根 目录 下 创建 一 个 HTML 文件 ， 并 命名 为 layouthtml。 该 文件 即 为 当前 
应 用 的 布局 文件 ， 使 用 可 视 化 工具 (如 Dreamweaver) 设计 一 个 网 站 外 观 ， 将 需要 经 党 改动 
的 代码 使 用 {__ CONTENT __“} 特 定 变量 〈 没 有 定 界 符 ) 代 伏 ， 如 以 下 代码 所 示 。 


<html> 
<head> 


content side 


<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
<meta name="description" content="" /> 
<meta name="generator" content="HTML-Kit" /> 
<ricle><l—— {Strit le}==></tit le> 
<import type='css' file="Css.main" /> 
</head> 
<body> 
<div id="outside"> 
el ne Meo a ee TI < ln 
<h1> 欢 迎 光 I 临 <span style="color: #666666;"> 我 的 博客 </span></n1L> 
有 
</a> |<a 
href="#"> 发 表 日 志 </a></div> 
SE LO ELM eher) 
<ul class="ul-menu"> 
<li- <span class "slide sraguo, span a hrer POR Ja /> 
<li- Da cliss "lidet ead /span a hrer AR a 


-4 


< <span class "slide ad /span <a hrer bus Bä e Vë das d Li 


</ul> 
ERR ER e D Ee E 
<h4> 网 站 公告 </h4> 
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<P> 我 的 博 开 通 了 ， 欢 迎 来 访问 我 的 博 ! 我 将 在 这 里 对 总 结 我 的 学 习 成 果 ， 不 定时 地 发 表 一 些 文章 或 看 法 ， 和 希望 能 
够 与 各 位 共同 探讨 ， 共 同 进步 。</p> 

</div> 

<div class="info"> 

<h4> 网 站 留言 </h4> 

<p> 和 暂时 没有 留言 </p> 

</div> 

</div> 

<div id="main">{ CONTERN }</div> 

SEET EE | ee VN 
</a>|<a 


eet Mo > EE EE EH DE 
<div id="footer"> 
<h5>Copyright by me and not one other person 2012</h5> 
</div> 
</div> 
</body> 
</html> 


W ERRITAR R ERK CONTENT 17 E RAN AA IFE AS E OC IT, 
整个 解释 流程 将 由 layout.html 开始 ， 然 后 到 各 目的 动作 页 面 。 这 样 一 来 ， 我 们 只 需要 在 控制 
髓 动作 中 藤 入 少量 的 代码 ， 即 可 使 页 面 保 持 统一 的 布局 与 风格 ， 如 以 下 代码 所 未 。 


<taglib name="tp My" /> 
<tp:editor id="textContent" uploadurl="/Public/editor_up" width="600"></tp:editor> 


<form id="form1" name="form1" method="post" action=""> 
<screkxtarea name= VcontentY EE 
</form> 


在 用 户 访问 该 页 面 时 ， 上 述 代 人 码 将 会 代替 {_CONTENT _ } 变 量 ， 效 果 如 图 6-5 所 示 。 


ap 


kant le lf ` | | | 
DEET ae 
HTM ée se t Se Le te ka a "G BZ v e g ma a 
JE Gu SR nm H F m Ẹ E 7 U à e | 相遇 e 
ES 3 Re SS a a" 下 ei: ] e 


ES E 
EA | TRGA | EPEK | AE var, 


EA RES HOA ARS OLENS TIrAr Eraro DENG | 
图 6-5 网 站 布局 效果 
在 布局 文件 中 髓 入 的 变量 或 标签 等 都 是 非 全 局 的 ， 而 只 针对 当前 页 面 生 效 。 所 以 在 布局 
代码 中 声明 变量 或 语句 ， 都 应 该 在 当前 控制 器 动作 中 处 理 ， 否 则 就 没有 存在 的 意义 。 布 局 文 
件 允 许 葵 套 ， 如 果 网 站 页 面 更 加 复杂 ， 还 可 以 结合 include 等 标签 实现 开发 需求 。 
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另外 ， 如 果 不 需 要 全 局 布局 ， 也 可 以 单独 在 当前 页 面 中 使 用 布局 文件 ， 只 需要 在 当前 页 
面 头 部 加 上 <layout name=' 布 局 文件 名 称 ' /> 标签 即 可 ， 这 种 方式 称 为 局 部 布局 ， 事 实 上 与 
include 标签 原理 一 样 。 


6.5 小结 


本 章 由 浅 入 深 全 面 介绍 了 ThinkPHP 开发 MVC 的 步骤 。 通 过 本 章 学 习 ， 可 以 感受 到 
MVC 开发 的 高 效 和 灵活 。 本 章 重 点 介绍 了 ThinkPHP 的 路 由 模式 与 视图 引擎 ， 它 是 MVC 与 
用 户 打 交道 的 窗口 ， 理 解 并 掌握 这 些 内 容 是 设计 好 一 个 网 站 的 前 提 。 结 合 示例 代码 ， 读 者 应 
该 能 够 轻松 掌握 。 

为 了 便于 讲解 ， 本 章 并 没有 触及 数据 库 操作 知识 。 事 实 上 ， 数 据 库 操 作 才 是 一 个 网 站 的 
灵魂 ，ThinkPHP 提供 的 CURD 数据 库 操作 接口 让 这 一 切 变 得 容易 ， 下 一 章 将 会 深入 讲解 
MVC 数据 库 开 发 。 
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内 容 提 要 


开发 网 站 离 不 开 数 据 库 ，PHP 本 身 就 内 置 了 众多 数据 库 驱 动 ， 开 发 人 员 在 不 利用 第 三 
方 驱动 的 情况 下 也 能 够 轻松 地 使 用 PHP 进行 数据 库 系 统 开 发 。ThinkPHP 对 数据 库 操作 进 
行 了 高 度 封 装 ， 能 够 对 当前 所 有 主流 的 数据 库 提 供 高 效 的 编程 模型 ， 开 发 人 员 甚 至 不 需要 
了 解 当 前 数据 库 查 询 语 句 ， 也 能 够 使 用 面向 对 疹 的 CURD 封装 类 ， 灵 活 快捷 地 完成 数据 库 
系统 开发 。 

与 其 他 主流 MVC 框架 有 所 不 同 ，ThinkPHP 不 是 简单 地 重 写 PDO 类 (当然 也 支持 
PDO )， 而 是 所 有 数据 库 操 作 均 继承 于 Db 类 ， 该 类 针对 配置 文件 指定 的 配置 信息 ， 有 选择 地 
载 入 数据 库 驱 动 ， 默 认 使 用 MySQL 数据 库 驱 动 ， 其 他 数据 库 驱 动 系统 并 没有 自 带 (2.x 时 
自 带 )， 需 要 读者 自行 到 官方 网 站 下 载 。 在 操作 层面 ， 开 发 者 不 需要 担心 数据 库 间 的 差异 
性 ， 因 为 这 些 过 程 都 是 抽象 化 的 。 

本 章 将 对 数据 库 常见 操作 进行 深入 介绍 ， 包 括 数据 增 、 删 、 改 、 查 
简单 的 数据 库 操 作 ， 还 将 进一步 学 习 ThinkPHP 提供 的 模型 关联 、 高 性 


学 习 目 标 


掌握 数据 表 模 型 定义 。 

掌握 数据 库 CURD 操作 。 

掌握 AdvModel 高 级 模型 的 使 用 。 
了 解数 据 库 分 布 式 蜗 性 能 开发 。 

掌握 MySQL 数据 表 分 区 技术 。 


， 简 称 CURD. Wit 
能 查询 等 知识 。 
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7.1 定义 数据 表 模 型 


要 操作 数据 库 ， 首 先 需要 连接 数据 库 ，ThinkPHP 不 需要 手动 连接 数据 库 ， 而 是 由 系统 
自动 连接 或 释放 ， 读 者 只 需要 根据 5.3.1 节 内 容 配置 好 信息 ， 这 样 就 完成 了 连接 数据 库 所 需 
要 的 步骤。 接 下 来 就 可 以 针对 特定 的 数据 表 进 行 数据 库 系统 的 开发 了 。 


7.1.1 模型 映射 


要 测试 数据 库 是 否 已 经 正常 连接 ， 最 直接 的 办 法 束 是 在 当前 控制 器 动作 中 实例 化 数据 
表 ， 然 后 使 用 dump 函数 和 输出， 得 看 数据 库 的 连接 状态 ， 代 人 码 如 下 上 所 示 。 

public function hello(){ 

$obj=M ("User"); 
dump ($300); 

} 

然后 通过 URL 访问 hello 动作 ， 如 果 出 现 错 误 提 示 则 说 明 数 据 库 没 有 连接 上 确保 打开 
调试 模式 )， 需 要 重 狐 检查 配置 文件 。 盏 则 即 为 成 功 状 态 。 

上 述 代码 中 M(“User”) 即 为 模型 映射 ，M 函数 是 new Model0 的 快捷 方式 ，Model 类 是 模 
型 的 基 类 ， 也 是 数据 库 操 作 的 基 类 ， 访 类 封 帮 了 所 有 第 见 数 据 库 操 作 方法 ， 其 中 $name WiZ 
类 中 的 一 个 成 员 属 性 ， 表 示 模 型 名 称 ， 模 型 名 称 与 数据 库 中 的 数据 表 进 行 映射 ， 所 以 Model 
类 将 $name 作为 构造 函数 参数 传 入 ， 达 到 初始 化 的 目的 。 

在 模型 与 表 映 射 过 程 中 ， 系 统 会 智能 地 根据 配置 信息 处 理 好 模型 名 与 表 名 之 间 的 关系 。 
例如 上 述 代码 中 的 User 表 名 ， 事 实 上 数据 库 中 并 不 存在 User 表 ， 但 是 系统 会 根据 配置 信息 
DB_PREFIX 指定 的 数据 表 名 前 级 智能 地 添加 到 模型 中 ， 因 为 User 首 字母 为 大 写 ， 系 统 会 强 
制 给 User is Ju mu ER, WAKRA D tpk_user。 

PA E Y A AA RE E I Be He A DK R En AR K S R o A i Ae i K A N 
tp_user_local， 那 么 在 模型 映射 时 只 需 将 表 名 与 后 缀 名 首 字 母 改 成 大 写 即 可 。 

public function hello(){ 


Şobj=M ("UserLocal"); 
dump ($obj); 


} 

如 有 果 不 需 要 为 表 添 加 表 前 缀 ， 那 么 可 以 将 模型 名 称 首 字母 改 为 小 写 ， 即 M("'userLocal")。 
当然 在 实际 应 用 开发 中 是 极 少 这 样 设计 的 (如 果 全 部 都 不 需要 表 前 级 ，DB_PREFIX 配置 项 
中 留 空 即 可 )。 


7.1.2” 自 定义 模型 


Model 类 提供 了 一 些 基础 的 数据 库 操作 方法 ， 当 需要 进行 复杂 的 数据 库 操作 时 ， 就 需要 
使 用 自 定义 模型 了 。 自 定义 模型 能 够 实现 更 加 灵活 和 强大 的 数据 库 操作 ， 特 别 是 能 够 针对 项 
目 一 些 特殊 的 功能 定制 数据 库 操 作 方 案 ， 例 如 数据 检验 、 数 据 缓存 、 数 据 加 工 等 。 使 用 自 定 
义 模型 ， 通 常情 况 下 也 是 需要 与 数据 库 的 真实 表 名 进行 映射 ， 这 是 MVC 的 一 条 规范 ， 否 则 
模型 就 与 普通 的 功能 类 无 异 。 在 调用 时 ， 系 统 提 供 了 D 函数 ， 用 于 快速 实例 化 自 定 义 模型 。 
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下 面 将 结合 示例 代码 介绍 目 定 义 模 型 的 创建 和 使 用 。 

1. 创建 目 定 义 模 型 

日 定义 模型 存放 于 项 目 Lib/Model 目录 下 ， 假 设 需要 为 数据 表 tpk_article 数据 表 建 立 模 
型 映射 ， 那 么 需要 创建 ArticleModelclass.php 文件 。 创 建 完 成 后 ， 在 使 用 D 函数 进行 实例 化 
时 ，ArticleModel 模型 将 与 tpk_article 表 进 行 映 射 。 由 于 所 有 模型 都 需要 继承 于 Model 基 
类 ， 所 以 ArticleModel 拥有 Model 所 有 特性 ， 这 里 将 使 用 select0 方 法 输出 tpk_article 表 数 
据 ， 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 
Buolie funmetion arciele() i 
Sobj=D ("Article"); 
Şrows=$0obj->select () ; 
dump ($rows); 
i 
| 


使 用 自 定义 模型 之 后 ， 在 进行 数据 表 操作 时 就 更 加 灵活 。 比 如 需要 让 tpk article 表 的 内 
容 能 够 根据 客户 所 在 的 地 区 显示 当地 的 新 闻 ， 那 么 就 可 以 在 ArticleModel 模型 中 对 数据 进行 
加 工 和 处 理 ， 如 以 下 代码 所 示 。 


<?php 
class ArticleModel extends Model{ 
public function article(){ 


Şrows=$this->where ("area ='{$this->checkUserArea()}'")->select(); 
return S$rows; 

} 

/大 大 

* 检测 用 户 所 在 城市 

s Enter description bere soc 

E 

protected function checkUserArea (){ 
import ("ORG.Net.IpLocation"); 
Smyio = ger cliene ipl) y 
ŞIp = new IpLocation ("UTFWry.dat"); 
Şarea = Ş$Ip->getlocation($myip); 


return $area; 


} 


上 述 代 码 中 ， 使 用 了 $this->where 连贯 操作 ， 访 语句 对 SQL 查询 语句 中 的 where 进行 了 
封装 ， 实 际 上 效果 是 等 同 的 。 在 where 查询 条 件 中 ， 限 制 返回 


字段 类 型 
数据 条 件 为 当前 地 区 的 新 闻 ，checkUserArea 是 一 个 日 定义 方 id tinyint(4) 
法 ， 用 于 根据 客户 端 全 ， 自 动 取得 当前 用 户 所 在 的 区 域 ， 作 为 DS 

E A 、 F] content text 
一 个 功能 方法 不 需要 对 外 公开 ， 所 以 修饰 为 protected 。 E 
ArticleModel 模型 对 应 的 tpk_ article 数据 表 结 构 如 图 7-1 所 示 。 Se char(32) 
2. 使 用 自 定 义 模型 add user char(24) 
add time int(9) 


使 用 目 定 义 模 型 非常 价 单 ， 系 统 提供 了 D 函数 ， 用 于 快速 
实例 化 目 定 义 模型 。 以 前 面 创建 的 示例 为 例 ， 在 动作 中 只 需要 图 7-1 tpk_article 表 结构 
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调用 article 方法 即 可 ， 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 
public function article(){ 
SE GA e EE 
Şrows=$0obj->als(); 
Şthis->assign("list",$rows); 
şthis--display(); 
| 
i 


XER EHAR A ENRE AEREE. AAE, EH A ENER Sr 
据 表 比 直 接 使 用 基 类 的 模型 映射 更 加 灵活 和 强大 。 事 实 上 这 种 方式 在 主流 的 MVC 中 使 用 得 
最 广泛 ， 其 至 是 必须 这 样 做 ， 例 如 前 面 草 和 介绍 过 的 Zend Framework, Symfony, CakePHP 
等 。 使 用 目 定 义 模型 能 够 实现 更 复 林 及 更 高 级 的 功能 ， 这 里 只 是 简单 地 演示 其 使 用 步 又 ， 在 
实际 应 用 开发 中 可 以 根据 需求 利用 目 定 义 模型 对 数据 进行 深度 的 加 工 、 处 理 等 。 

自 定 义 模型 映射 ， 一 个 模型 对 应 一 个 数据 表 。 所 有 增 、 删 、 改 、 碍 都 在 模型 类 中 完成 ， 
大 大 地 方便 了 文件 及 代码 的 管理 。 目 定义 模型 的 最 大 特点 是 代码 容易 移植 ， 高 度 重 用 。 例 如 
一 个 客户 端 只 能 访问 XML 格式 数据 ， 此 时 目 定义 模型 一 名 代码 都 不 用 更 改 ， 只 需要 改变 动 
作 中 的 输出 格式 即 可 。 

这 里 需要 说 明 的 是 ， 在 ThinkPHP 中 模型 代码 是 可 以 直接 移植 到 动作 中 的 ， 但 这 样 束 让 
代码 的 重复 利用 变 差 ， 代 码 的 管理 、 维 护 变 得 困难 ， 上 所 以 建议 读者 尽量 将 数据 库 操 作 的 逻辑 
PSA ENAR, XE MVC 编程 所 提倡 的 思想 。 


7.1.3 create 方法 


表面 多 次 出 现 的 select 方法 为 Model 类 的 一 个 成 员 方 法 ， 该 方法 用 于 列 出 所 有 符合 条 件 
的 数据 。 在 数据 库 开 发 中 ， 碍 询 和 插入 数据 是 同等 重要 的 ，ThinkPHP 对 数据 的 插入 、 更 新 
等 都 做 了 高 度 封 装 ， 系 统 提 供 create 方法 ， 用 于 创建 数据 对 象 。 

所 谓 的 数据 对 象 束 数据 字段 与 数据 表 之 间 的 关系 ， 数 据 会 被 映射 为 类 成 员 ， 然 后 再 与 数 
据 表 映 射 ， 最 后 实现 数据 的 插入 或 更 新 。 对 开发 者 而 言 ， 这 一 过 程 不 需要 关心 ， 只 需要 在 表 
单 中 设置 好 字段 〈 表 单元 素 )， 系 统 会 目 动 建立 好 映射 天 系 。create 方法 是 ThinkPHP 最 基础 
和 最 强大 的 数据 操作 方法 ， 该 方法 是 连贯 操作 、CURD 操作 的 集合 ， 包 括 了 数据 创建 、 数 据 
检验 、 表 单 验证 、 上 自动 完成 等 实用 功能 ， 接 下 来 将 详细 介绍 create 方法 。 

1. create 创建 数据 流程 

Create 的 数据 源 由 Post 表单 提供 ， 一 般 情 况 下 开发 者 不 需要 做 任何 的 更 改 ， 即 可 让 表单 
元 素 中 的 数据 目 动 映射 为 数据 表 中 的 数据 。 例 如 表单 中 有 username KAR, MAIZE 
会 目 动 被 映射 为 数据 表 中 的 username 字段 ， 在 数据 创建 的 过 程 中 ，create 方法 会 目 动 对 数据 
进行 处 理 ， 确 你 数据 的 安全 和 有 效 。 过 程 如 图 7-2 所 示 。 

数据 对 和 象 创建 成 功 后 ， 访 对象 被 存放 于 内 存 中 ， 此 时 可 以 使 用 Model 类 中 的 add 方法 对 
数据 进行 保存 或 更 新 操作 (create 方法 自动 判断 是 保存 还 是 更 新 )。 
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表单 数据 


是 否 通过 数据 检验 ? 


过 渡 表 单数 据 


数据 创建 完成 


生成 数据 对 象 ， 
保存 在 内 存 中 


图 7-2 create 创建 数据 流程 


2. create 数据 操作 


前 面 介 绍 了 create 方法 的 运行 过 程 ， 读 者 只 需要 理解 即 可 ， 在 实际 应 用 开发 中 创建 数据 
JJ 
mj 


的 过 程 是 极其 简单 的 。 下 和 耐 将 结合 示例 代码 ， 介 
法 的 认识 。 


create 方法 的 实际 应 用 ， 加 深 对 create Jr 


首先 在 Index 控制 右 中 创建 一 个 用 于 接受 用 户 输入 的 表单 页 面 ， 这 里 将 控制 句 动 作 命名 


为 add_article， 人 代码 如 下 所 示 。 


<?php 
class IndexAction extends Action { 
public function add_article(){ 
Şthis->display ();}; 
1 
} 
?> 


add article 动作 对 应 的 add_article html 页 面 代 码 如 下 所 示 。 


<taglib name="tp" /> 


<form method="post" action=" __URL_ /add"> 
过 全 ve nane eiren EE 
<- Ce inpute type eet" nane add nser" id add user value HE 
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<1i> 地 区 : 
<label> 
<select name="area" id="select"> 
<option value="guangdong"> 广 东 新 闻 </option> 
<option value="beijing"> 北 京 新 闻 </option> 
<option value="shanghai"> 上 海 新 闻 </option> 
<option value="chongqing"> 和 草 庆 新 闻 </option> 
</select> 
</label> 
</i> 
<1i> 分 类 : 
<label> 
<select name="category" id="category"> 
<option value="1"> 和 社会 民生 </option> 
<option value="2"> 体 育 新 闻 </option> 
<option value="3"> 国 际 军 事 </option> 
<option value="4"> 财 经 动态 </option> 
</select> 
</label> 
</> 
<li> 
<textarea name="content" id="textContent" cols="45" rows="5"></textarea> 
</i> 
<div class="addSubmit"> 
<input Ee submit vilie We 
</div> 
</form> 
<tp:editor id="textContent" uploadurl="/Public/editor_up" width="600"></tp:editor> 
<style> 
li{ 
list-style:none; 


margin: 6px; 


| 

.addSubmit { 
text-align:center; 
width: 600px; 

} 

.addSubmit input { 
width:100px; 

Tont =SL26e3 he Ee 
height :32px; 

} 

</style> 

<div> 

</div> 


在 上 述 代 人 码 中 ， 在 表单 中 共 定 义 了 5 个 字段 ， 分 别 为 title, add user, area, category, 
content。 其 中 area, category 使 用 了 表单 列表 元 素 ，content 使 用 了 pE 内 置 的 富 文 本 
编辑 器 ， 所 以 需要 使 用 staglib name="tp" 人 额外 引入 tp 扩展 标签 类 库 ， 最 终 呈现 效果 如 图 7-3 
所 示 。 

在 上 述 代 人 码 中 ， 表 香 提 交 页 面 为 _URL_/add， 即 当 胡 控制 右 中 的 add 动作 。 接 下 来 的 
BAEIJE add 动作 中 人 完成， 如 以 下 代 人 码 所 示 。 
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ZU l 7 G 我 的 博客 e Di English 


HT | RES | 美 于 二 站 | AAR | TR | 


图 7-3 add article.html 页 面 效果 


<?php 
class IndexAction extends Action { 
// 表 单 处 理 
public function adqdq() { 
Şarticle0bj = M('Article'); 
$article0bj->create(); 
Şarticle0bj->add_time=time (); 
i ($arcıicle0bj=>acel()) A 
Sthis->success (" 数 据 添 加 成 功 ") ; 
}else{ 
$this->error (" 数 据 添加 失败 ") ; 
} 
} 
?> 
如 上 述 代码 所 示 ， 使 用 M 函数 直接 实例 化 数据 表 tpk_article， 读 者 也 可 以 在 日 定义 模型 
中 完成 。 然 后 使 用 create 方法 创建 数据 ， 创 建 的 过 程 不 需要 开发 者 手动 赋值 ， 系 统 会 再 接 将 
表单 中 的 表 蛙 元 素 直 接 与 tpk_article 数据 表 进 行 遇 射 。 但 是 ， 由 于 表单 中 没有 add_time 元 
率 ， 这 里 可 以 直接 以 对 象 的 方式 为 articleObj 数据 对 象 添加 类 成 员 。 最 后 的 结果 束 是 添加 一 
个 add_time 表 字 段 ， 该 字段 的 值 为 当前 时 间 惟 。 一 切 完成 后 ， 此 时 数据 对 象 存放 于 内 存 中 ， 
使 用 add 方法 进行 提交 ， 完 成 整个 create 创建 数据 的 过 程 。 


7.1.4 模型 属性 


通过 在 目 定 义 模型 中 定义 其 成 员 属 性 ， 可 以 更 改 模型 与 数据 表 的 属性 ， 提 高 目 定 义 模型 
的 灵活 性 。 在 目 定 义 模型 中 ， 第 见 的 成 员 属 性 包括 fields、tableName、dbName、_map 等 ， 
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这 些 属性 都 必须 修饰 为 protected 权限 ， 下 面 分 别 介绍 。 

1. fields Æl (RFF) 

在 模型 里 可 以 定义 fields 属性 ， 访 属性 的 值 为 一 个 二 维 数组 ， 数 组 元 素 会 被 系统 映射 为 
数据 表 字 段 。 默 认 情 况 下 ， 在 第 一 次 初始 化 模型 时 ， 系 统 会 将 数据 表 字 段 写 入 到 缓存 文件 
中 ， 这 些 缓存 文件 名 称 与 数据 表 名 相同 ， 如 tpk_articLphp。 绥 存 数 据 表 字段 能 够 提高 数据 的 
吧 应 速度 ， 减 少 租 询 时 的 资源 消耗 。 但 是 默认 情况 下 ， 系 统 会 将 表 中 的 所 有 字段 进行 缓存 ， 
但 往往 我 们 只 需要 表 中 的 某 个 字段 ， 这 无 疑 会 造成 性 能 损失 。 另 外 ， 如 果 数 据 表 字段 过 多 ， 
在 进行 缓存 时 将 会 造成 极 大 的 IO 开销 ， 形 成 性 能 瓶 开 ， 尤 其 在 大 型 的 分 布 式 开发 中 更 是 需 
要 注意 。 在 fields 属性 中 进行 明文 定义 数据 表 字 段 ， 能 够 避免 系统 对 数据 表 字 段 进行 文件 组 
存 。 由 于 字段 是 明文 标识 的 ， 系 统 在 进行 数据 库 操 作 时 不 会 重复 与 数据 表 进 行 映 射 ， 而 是 与 
明文 定义 的 fields 属性 值 下 接 殴 射 ， 这 样 束 有 效 地 提高 了 数据 操作 效率 。 

定义 fields 属性 ， 需 要 在 日 定义 模型 中 进行 。 在 定义 模型 学 段 时 ， 系 统 会 默认 模型 字段 
添加 一 个 主键 id， 但 更 民 好 的 编程 习惯 是 显 式 指定 ， 开 发 人 员 可 以 使 用 _pk TRIR EEEF 
段 ， 并 使 用 _autoinc 元 系 指 定 目 动 增 长 ， 如 以 下 代码 所 示 。 

E ArticleModel extends Modell 

protected $fields = array ( 
nekte 
'content', 


'category', 
E TEE 
warclelml ne 
'_autoinc' => true, 
pki = nl) 

i 

e 


?> 


如 果 不 定 义 模 型 字段 ， 叉 不 需要 缓存 数据 表 字 段 ， 那 么 可 以 配置 DB_FIELDS_ 
CACHE'=>false 完全 关闭 字段 缓存 功能 。 如 果 关 闭 缓存 ， 又 不 在 目 定 义 模型 中 定义 fields 属 
性 ， 那 么 系统 每 次 调用 自 定 义 模 型 时 ， 都 会 往 数据 表 中 读 取 表 字 段 。 

2. tableName 属性 〈 表 映射 ) 

前 面 已 经 介绍 过 ， 模 型 的 命名 需要 与 数据 表 名 相同 ， 这 样 系统 才能 将 模型 与 数据 表 进 行 
映射 ， 否 则 在 调用 时 只 能 当成 一 个 普通 的 系统 类 使 用 。ThinkPHP 灵活 之 处 束 在 于 任何 的 操 
作 都 不 是 绝对 的 ， 定 义 模型 同样 也 是 。 系 统 允 许 开发 人 员 通 过 定义 tableName 成 员 属 性 改变 
当前 模型 与 数据 表 映 射 关系 ， 使 用 方式 也 比较 简单 ， 如 以 下 代码 所 示 。 


<?php 

class ArticleModel extends Model) 
protected StableName = 'news'; 
E 

1 

?> 


如 上 述 代 码 所 示 ， 通 过 定义 tableName 属性 ，Article Datum DI news KT- 
tableName 的 值 为 数据 表 名 ， 表 名 不 能 向 表 前 级 ， 不 区 分 大 小 写 。 需 要 注意 的 是 ， 如 果 定 义 
了 fields 属性 ， 再 使 用 tableName 更 改 数 据 表 名 ， 那 么 fields 中 的 字段 名 称 需 要 与 tableName 
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中 指定 的 表 字 段 名 称 相同 。 虽 然 tableName 能 够 改变 表 名 称 ， 但 在 实际 应 用 开发 中 应 该 一 个 
模型 对 应 一 个 数据 表 ， 尽 量 避 人 免 使 用 tableName 属性 。 

3. dbName 属性 〈 切 换 数据 库 ) 

dbName 非常 灵活 和 强大 ， 尤 其 在 超大 型 的 分 布 式 网 站 开发 中 ， 显 得 非常 有 用 。dbName 
能 够 让 当前 模型 在 不 牺牲 性 能 的 情况 下 迅速 切换 数据 库 ， 并 且 这 一 过 程 是 持续 的 ， 对 用 户 而 
言 不 需要 中 断 当 前 操作 。dbName 默认 情况 下 只 能 切换 同一 服务 占 的 数据 库 ， 如 采 配 合 分 布 
式 数 据 库 配置 ， 也 能 够 跨 数 据 库 服务 器 切换 。dbName 数据 库 切 换 流程 如 图 7-4 所 示 。 


tp 数据 库 
tpk_article 数 据 表 


是 否定 义 protected 
$dbName='tp2'? 


tp2 数 据 库 


生成 数据 对 象 tpk_article 数 据 表 


ICC 
图 7-4 在 自 定 义 模 型 中 切换 数据 库 


如 图 7-4 所 示 ， 通 常情 况 下 Article 模型 被 映射 成 tp 数据 库 中 的 tpk_article 数据 表 〈 根 
据 配 置 文件 中 所 配置 的 信息 而 定 )， 如 果 在 模型 中 定义 了 dbName 属性 ， 那 么 执行 流程 将 发 
生 改 变 ，Article 模型 将 会 被 映射 为 tp2 数据 库 中 的 tpk_article 数据 表 。 这 一 过 程 虽然 简单 ， 
但 在 实际 应 用 开发 中 能 够 有 效 提高 数据 服务 器 的 性 能 。 利 用 dbName 特性 ， 开 发 人 员 可 以 创 
建 两 个 表 结 构 完全 一 样 的 数据 库 ， 然 后 在 模型 中 动态 更 改 dbName 属性 的 值 ， 实 现 简 单 的 分 
布 式 开发 ， 比 如 当主 数据 库 中 的 tpk_article 数据 表 过 高 记录 时 超过 百 万 行 记录 )， 此 时 可 以 
按照 记录 条 数 比 例 动态 切换 到 副 数 据 库 ， 就 能 够 有 效 地 提高 数据 读 写 性 能 。 

切换 数据 库 后 ， 还 可 结合 tableName 一 起 使 用 ， 但 意义 并 不 大 ， 因 为 通常 切换 数据 库 只 
作为 提高 性 能 的 一 种 方式 ， 如 果 再 更 改 模型 映射 将 会 让 代码 维护 变 得 困难 。 利 用 dbName 特 
性 ， 在 开发 数据 库 热 备份 程序 时 将 变 得 容易 。 使 用 dbName 非常 简单 ， 只 需要 在 自 定义 模型 
中 定义 即 可 ， 代 人 码 如 下 。 


<?php 

class ArticleModel extends Model{ 
protected $dbName = 'tp2'; 
VURE E 


} 


4. map 属性 (字段 映射 ) 

前 面 已 经 介绍 过 使 用 create 方法 可 以 方便 地 创建 数据 ， 使 得 表单 与 数据 库 交互 变 得 容 
易 。 但 这 里 面 有 一 个 不 容 扳 视 的 问题 ， 存 HTML 表单 中 的 表单 元 素 必 须要 与 数据 表 中 的 字 
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段 名 称 相 同 ，create 方法 才能 正确 创建 数据 ， 这 束 意 味 着 数据 表 字 段 显 式 地 暴露 在 页 面 中 ， 
这 对 要 求 安全 严 紧 的 网 站 来 说 是 不 可 以 接受 的 。 虽 然 create 方法 已 经 对 数据 进行 了 严格 的 控 
制 ， 但 数据 表 字 段 暴露 在 外 始终 是 一 个 安全 隐患 。_map 属性 可 以 将 HTML 表单 中 的 元 素 以 
别名 的 方式 映射 到 真实 的 数据 表 字 段 ， 和 其 他 属性 一 样 ，_map 属性 也 必须 定义 在 上 自 定 义 模 
型 中 ， 该 属性 的 值 为 二 维 数组 ， 下 面 以 7.1.3 节 内 容 为 基础 ， 修 改 add_article.html 表单 元 素 
与 add 动作 代码 ， 并 在 Article 模型 中 创建 createAdd 方法 ， 用 于 处 理 表单 数据 。 
add_article.html 代码 如 下 所 示 。 


<taglib name="tp" /> 
<form method="post" action=" __URL_ /add"> 
lies <input type="text" name="subject" id="subject" value="" size="30"/> 
</i> 
<li>iFáø: <input type "text" name- "author" id "author" value wn size "30"/> 
过 /本 工交 
<1i> 地 区 : 
<label> 
—ëelect pname--jdietrciet" LOWELL ELET N 
<option value="guangdong"> 广 东 新 闻 </option> 
<option value="beijing"> 北 京 新 闻 </option> 
<option value="shanghai"> 上 海 新 闻 </option> 
<option value="chongqing"> 和 草 庆 新 闻 </option> 
</select> 
</label> 
SE 
AE ée 
<label> 
<select name="type" id="type"> 
<option value="1 "ESR Æ</option> 
<option value="2"> 体 育 新 闻 </option> 
<option value="3"n> 国 际 盏 事 </opticny> 
<option value="4"> 财 经 动态 </option> 
</select> 
</label> 
SE 
SE 


<textarea name="textEdit" id="textContent" cols="45" rows="5"></textarea> 

SS 

<div class="addSubmit"> 

input cyp vem le Von 

</div> 

</form> 

<tp:editor id="textContent" uploadurl="/Public/editor up" width="600"></tp:editor> 
<style> 

Lo 

list-style:none; 


margin: 6px; 


| 

.addSubmit { 
Lext-alionrcenter: 
width:600px;} 
.addSubmit input{ 
width:100px; 
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font-size:16px; 

height :32px; 

E 

如 上 述 代 码 所 示 ， 标 粗 的 即 为 需要 提交 的 表单 元 素 ， 这 些 元 素 名 称 已 经 发 生 了 改变 ， 与 
数据 表 tpk_article 中 的 字段 完全 不 相关 。 接 下 来 在 Article 模型 中 使 用 _map 属性 将 这 些 元 素 
DD D tpk_article 表 中 的 相应 学 段 。 代 人 码 如 下 所 示 。 


<?php 
class ArticleModel extends Model{ 
protected $_map=array ( 
EE ee 


"author"=>"add_user", 


vamser rieti tarea 
TYDS SS CATEGO"; 
"textEdit"=>"content" 


) 7 

/ KK 

* 处 理 表单 提交 数据 

” Barcer description bere ooo 

2 

public function createAdd()t{ 
ʻ$arcLele0oJ = EE ees 
ŞarticleO0bj->add_time=time (); 
return $this->add();}; 


} 


如 上 述 代码 所 示 ，_map 属性 的 值 为 天 联 数组 ， 其 中 键 为 表单 字段 名 称 ， 值 为 需要 映射 
的 数据 表 字 段 。 在 处 理 方 法 中 ， 只 需要 下 接 调用 父 类 (Model) 中 的 create 方法 即 可 创建 正 
确 的 数据 。 

在 表单 提交 处 理 动 作 中 ， 只 和 需要 和 直接 调用 目 定 义 模型 中 的 create Add 方法 即 可 ， 如 以 下 
代码 所 示 。 


<?php 
class IndexAction extends Action { 


public function add(){ 
Sarticleobj = D('Article'!); 
if ($articleObj->createAdd())t 
Sthis->success (" 数 据 提 交 成 功 ") ; 
}else{ 


$this->error ("数据 提交 失败 ")，; 


7.2 基础 模型 ( Model ) 


在 前 面 的 内 容 中 多 次 出 现 的 where, create, select 等 方法 就是 用 于 操作 数据 库 的 最 基础 
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方法 。ThinkPHP 为 了 提高 开发 效率 ， 在 以 create 方法 的 基础 上 派生 了 许多 功能 强大 、 简 单 
易 用 的 实用 方法 ， 开 发 人 员 使 用 这 些 方法 可 以 极 大 地 提高 开发 效率 。 这 些 操作 者 是 面 问 对 象 
的 ， 开 发 人 员 不 需要 写 传统 的 SQL 语言 ， 上 只 需要 调用 相应 的 方法 并 赋 给 正确 的 参数 即 可 ， 
系统 会 目 动 转换 成 对 应 数据 库 能 够 解释 的 语句 ， 下 面 首 先 从 连贯 操作 开始 讲解 。 


7.2.1 连贯 操作 

ThinkPHP 框架 使 用 _call 魔术 方法 巧妙 地 实现 了 数据 库 连 贯 操作 。 所 谓 的 连贯 操作 是 指 
人 允许 将 多 个 对 象 方法 串联 在 一 起 使 用 ， 从 而 构造 成 完整 的 SQL 语句 ， 对 开发 者 而 言 不 需要 
关心 SQL 语句 的 写法 ， 只 需要 正确 赋 参 即 可 ， 常 见 的 连贯 操作 方法 如 表 7-1 所 示 。 


表 7-1 连贯 操作 方法 


Ho ak 作 H 文 持 的 参数 或 参数 类 型 
where 条 件 限 制 String、 Array、Object 
table 操作 的 表 名 String, Array 
alias 定义 表 别 名 String 
data 新 增 或 更 新 时 押 需 要 的 数据 Array, Object 
field 显示 的 表 字 段 String、Array 
order 数据 排序 String, Array 
limit 限制 查询 条 数 String, Int 
group 对 得 询 分 组 的 文 持 String 
having 对 having 语句 的 文 持 String 
join 对 join 语句 的 文 持 String, Array 
union 对 union 语句 的 支持 String、Array、Object 
distinct 是 否 对 数据 去 重复 true 或 false 
lock 是 否 锁定 数据 库 true 或 false 
cache SQL A WAT String 


连贯 操作 在 使 用 方式 上 天 如 同 将 一 条 合法 的 SQL BA TRORA, FER 
统 也 是 根据 这 一 流程 进行 转换 的 。 例 如 以 下 SQL 语句 。 

SELECT * FROM ‘tpk_ article WHERE add user SE ORDER BY ‘add time DESC LIMIT 
© - 30 

WREE E, MARAE A, mu Fra, 

ŞarticleO0bj=M ("Article"); 

Şrows=$article0bj->where ("add _ user ='ceiba'")->order ("add_time desc")->limit (30)- 
>Belecr ()) 


读者 可 以 根据 表 7-1 的 连贯 方法 进行 串联 ， 使 用 连贯 方法 除了 select 方法 外 ， 其 他 的 连 
贯 方法 不 区 分 前 后 顺序 。 事 实 上 select 不 属于 连贯 方法 ， 该 方法 是 CURD 中 的 一 个 用 于 显示 
数据 的 方法 ， 表 示 显 示 所 有 数据 。 相 应 的 如 果 只 需要 显示 单条 数据 可 以 使 用 find 方法 。 


7.2.2 CURD 
在 应 用 程序 中 数据 库 的 操作 主要 分 为 4 大 类 : 创建 数据 (Create)、 更 新 数据 (Updata)、 
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读 取 数据 (Read) 及 删除 数据 (Delete)， 简 称 CURD。 主 流 的 MVC 框架 都 会 对 CURD 操 
作 进 行 封 装 ， 以 达到 易 用 、 灵 活 的 目的 。CURD 是 Web 开发 的 重点 ， 一 套 MVC 成 不 成 熟 、 
稳 不 稳定 很 大 程度 上 取决 于 CURD 处 理 方 式 。ThinkPHP 的 CURD 操作 是 以 连贯 操作 为 基础 
的 ， 通 常情 况 下 这 两 者 会 配合 使 用 。 下 面 分 别 进行 介绍 。 

1. 创建 数据 Cadd) 

前 面 已 经 介绍 过 功能 强大 的 create 方法 ， 在 CURD 操作 中 细 化 了 create 方法 的 步骤 ， 但 
底层 处 理 机 是 相同 的 。 创 建 数据 可 以 使 用 add 方法 ， 该 方法 接收 一 个 关联 数组 作为 参数 传 入 
( 键 值 对 )， 最 终生 成 的 数据 对 象 和 表 中 的 结构 是 一 样 的 。 如 以 下 代码 所 示 。 


<?php 


class IndexAction extends Action { 
OA 
public function post () { 
SarticleübizchM(iArticleini: 
Ee ee Eeer 
sdatal"add _ user"]=$_POST["author"]; 
Scara l Varsat ]=$_POST | v celLst ELGEN] $ 
adara ee ty E Ee 
$data["content"]=$_POST[|"textEdit"]; 
$data["add_time"]=time (); 
if (Ş$articleO0bj->add($data)) { 
$this->success ("数据 添加 成 功 ")， 
}else { 
$this->error ("数据 添加 失败 ")， 
} 
} 
} 


> 


可 以 看 到 ， 与 create 方法 相 比 ， 在 数据 细 化 方面 ， 开 及 人 员 需 要 于 动 创 建 数组 ， 并 且 将 
创建 的 数组 作为 参数 传 入 add 方法 。add 方法 是 Model 类 有 的 一 个 实例 方法 ， 该 方法 能 够 对 参 
数 中 的 数组 信息 进行 过 滤 、 转 换 ， 最 终 提交 给 DB 类 中 的 insert 方法 ， 完 成 数据 的 插入 。add 
方法 处 理 数 据 是 可 徘 、 安 全 的 ， 所 以 束 算 开发 人 员 不 对 数据 进行 处 理 ， 这 些 数据 进入 数据 表 
时 都 是 安全 的 。add 方法 还 可 以 结合 data 连贯 操作 方法 一 起 使 用 ， 如 以 下 代码 所 示 。 


<?php 


class IndexAction extends Action { 

J 

public function post (){ 
ŞarticleO0bj=M ("Article"); 
Scarcal Y eirceletj=$_POST | Y subject” ] 7 
sdatal"add user"|=$ POST["author"]; 
See Klee ele 
ER 
Geen Lee EE eeler e 
$data["add_time"]=time(); 
if (ŞarticleO0Obj->data (Ş$data)->add()) { 

Sthis->success ("数据 添加 成 功 ")， 

}else { 


$this->error ("数据 添加 失败 ")，; 
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} 


EK 


效果 和 使 用 参数 传递 是 一 样 的 。 上 述 示 例 中 数组 信息 是 和 tpk_article 表 结 构 是 一 样 的 ， 
在 实际 应 用 开发 中 需要 根据 需求 进行 更 改 。 

2. 更 新 数据 (save) 

可 以 创建 数据 ， 当 然 也 能 够 更 新 数据 。 在 ThinkPHP 中 对 数据 的 更 新 基本 上 和 创建 数据 
一 样 。 开 发 人 员 不 需要 针对 数据 库 编写 set 操作 ， 只 需要 使 用 save 方法 即 可 。 继 续 以 
tpk_article 为 例 ， 如 果 需 要 更 狐 id 为 2 的 记录 ， 那 么 只 需要 在 save 参数 中 指定 id 即 可 ， 如 
以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 
EE 
public function post () { 
ŞarticleO0bj=M ("Article"); 
$data["id"]=2; 
Scacal Yercle”j=$_ POST Ysubject" ] z 
$data["add_user"]=$_POST["author"]; 
ee EE SEA E E kees 
EE eve 
ee l Veconcenrc E e DE GE ben 
$data["add_time"]=time (); 
if (ŞarticleObj->save(Ş$data)) { 
Sthis->success ("数据 修改 成 功 ")， 
}else { 


$this -error (E Or Drun. 


} 
i 
} 


?> 


当然 ， 在 实际 应 用 开发 中 id 的 值 应 该 取 目 于 $_POST 或 者 $_GET。save 的 使 用 方式 和 
add 基本 一 样 ， 为 了 数据 表 安 全 ， 系 统 不 允许 提交 空 更 新 条 件 的 操作 ， 如 上 述 代 码 中 的 
$data["id"] 即 为 更 新 条 件 ，id 为 操作 主键 ， 在 没有 条 件 时 均 可 作为 操作 条 件 。 如 果 数 据 表 或 
更 新 条 件 不 以 id 作为 主键 ， 那 么 可 以 配合 where 连贯 操作 来 限定 条 件 ， 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 
EE 
public function post (){ 
SarticleübizchM(iArticleini: 
Ee Ee 
sdatal[l"adqd user"]=$ POST["author"]; 
Ee Kee EE les 
odaca Caeeoor ya | PO TE ype] 
$data["content"]=$_POST[|"textEdit"]; 
$data["add_time"]=time(); 


if ($articleObj->where ("id=2")->save ($data)) { 
sthis->success ("数据 修改 成 功 "); 
}else { 


$this->error ("数据 修改 失败 ")， 
} 
} 
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} 


以 上 是 save 方法 的 使 用 ， 在 更 新 操作 中 系统 还 提供 了 针对 不 同 更 新 类 型 的 快捷 方法 。 
如 下 所 未 。 

> setField( 表 字段 , 值 ): 更 新 单个 字段 的 值 ， 需 要 配合 where 连 员 操作 使 用 。 

> setImnc( 字 段 , 增 加 值 ): 对 表 中 的 特定 字段 增加 数值 ， 该 字段 的 类 型 必须 为 int 类 型 。 

需要 配合 where 连贯 操作 使 用 。 

> setLazylne CE MIME, TE: 文 持 延迟 更 狐 的 setnc， 时 间 以 秒 为 单位 。 

> setDec( 学 段 , 减 少 值 ): 对 表 中 的 特定 字段 减少 数值 ， 该 字段 的 类 型 必须 为 int 类 型 。 

需要 配合 where 连贯 操作 使 用 。 

> SetLazyDec( 字 段 , 减 少 值 ,时 间 ): 支持 延迟 更 狐 的 setDec， 时 间 以 秒 为 单位 。 

3. 读 取 数据 (select) 

在 前 面 的 内 容 中 已 经 多 次 接触 过 select 方法 ， 访 方法 为 CURD 操作 中 最 常用 的 方法 。 
select 方法 返回 的 是 二 维 数组 ， 本 喘 并 不 会 格式 化 数据 ， 需 要 在 视图 中 使 用 volist 循环 标签 进 
行 处 理 。 如 果 需 要 返回 单条 数据 可 以 配合 limit 连贯 操作 ， 或 者 使 用 find 方法 (后 者 不 完全 
兼容 volist 标签 );。 由 于 它 的 使 用 方式 在 前 面 的 内 容 中 己 经 多 次 出 现 ， 在 此 束 不 再 细 述 。 这 里 
需要 注意 的 是 ，select 方法 默认 情况 下 会 将 数据 表 中 的 字段 全 部 输出 ， 这 会 造成 资源 浪费 ， 
解决 的 办 法 是 配合 getField 连贯 操作 ， 如 以 下 代码 所 示 。 

Ee IndexAction extends Action { 

EE 
public function index(){ 


ŞarticleO0bj=M ("Article"); 
Şrows=$article0bj->getField("id,title,content")->select(); 


Şthis->assign("list",$rows); 
Seles -display Oy 
} 

} 


?> 


4. 删除 数据 (delete) 

delete 方法 可 以 方便 地 删除 数据 ，delete 配合 where 连贯 操作 可 以 根据 条 件 进行 删除 操 
作 ， 如 果 配 合 limit 还 可 以 实现 人 窗 单 的 批量 删除 。 当 然 也 可 以 配合 PHP 中 的 foreach 或 者 for 
语句 进行 批量 删除 ，delete 的 使 用 方式 和 select 基本 一 样 ， 这 里 就 不 作 过 多 介绍 ， 下 面 将 以 
代码 演示 delete 方法 的 使 用 。 


<?php 
class IndexAction extends Action { 
E 
publie fumetion celete() i 
ŞarticleO0bj=M ("Article"); 
if (SarticleOb]j->where ("id=".Ş$_GET["id"])->delete()) { 
Sthis->success (" 删 除 成 功 " 
yelse { 
Sthis->success ("删除 失败 ")， 


x 


~ 
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如 朱 数 据 表 的 主键 为 ia， 并 且 以 id 作为 删除 条 件 ， 那 么 还 可 以 二 接 在 delete 方法 中 传 入 
id 参数 ， 如 以 下 代 但 所 示 。 


ŞarticleO0bj=M ("Article"); 
$article0bj ->delete (1); // 删除 主键 为 1 的 数据 
$articleObj ->delete('2,3'); // 删除 主键 为 2、3 的 多 个 数据 


7.2.3 ”查询 语言 


剖面 已 经 介绍 过 在 连 员 操作 中 的 whrere 方法 ， 访 方法 为 查询 限制 条 件 ， 接 受 的 参数 为 
SQL 但 询 条 件 部 分 ( 即 where 部 分 语句 )， 在 赋 参 时 一 般 使 用 字符 串 作为 但 询 条 件 ， 如 
where(“add_user=’ceiba””)。 事 实 上 这 种 方式 是 不 建议 使 用 的 ， 因 为 这 会 叶 斤 查询 洲 出 (SQL 
注入 、 查 询 边 界 超出 等) 造成 安全 隐患 。 所 以 系统 提供 了 多 套 人 查询 方式 ， 其 中 最 常用 的 为 数 
组 查询 方式 ， 使 用 数组 代 巷 字符 串 ， 系 统 会 对 有 所 有 数据 进行 过 小 ， 确 保 数 据 最 终 的 安全 。 数 
组 查询 方式 能 够 实现 大 部 分 字符 串 所 能 够 实现 的 查询 类 型 ， 包 括 常 见 的 普通 查询 、 区 间 查 
询 、 组 合 人 查询 、 统 计 查 询 等 。 使 用 数组 进行 查询 操作 ， 束 需要 使 用 查询 表达 式 ， 否 则 传统 的 
学 从 串 表 达 式 将 不 能 正确 解释， 下 和 面 首先 理解 查询 表达 式 。 

1. 查询 表达 式 

SQL 语言 是 一 种 比较 类 似 人 类 目 然 用 语 的 结构 化 查询 语言 ， 它 使 用 的 表达 式 与 传统 的 数 
学 表达 式 非 常 类 似 。 和 常见 的 等 号 (=)、 不 每 号 (<>)、 大 于 (>)、 小 于 (<) 等 表达 式 都 通 
俗 易 恒 ， 一 般 的 开发 人 员 几 平 不 必 牢 记 ， 也 能 运用 娴熟 。 这 是 SQL 语言 的 一 大 特性 ， 但 是 
这 些 表达 式 的 赋值 是 有 讲究 的 ， 如 末 处 理 不 当 就 会 造成 严 草 的 安全 隐患 ， 早 些 年 非常 普 授 的 
SQL 注入 百 分 之 九 十 都 出 于 这 些 表达 式 的 运用 不 当 。 面 对 这 些 问题 ， 在 CH Java 等 主流 语 
言 中 使 用 预 处 理 机 制 能 够 帮助 提高 查询 的 安全 性 ， 但 PHP 是 一 种 脚本 语言 ， 必 须 依 徘 开 发 
人 员 使 用 过 滤 函 数 进 行 处 理 ， 如 果 每 个 赋值 都 手动 过 滤 ， 这 无 疑 是 效率 低下 的 。 

在 MVC 开发 的 年 代 ， 这 一 切 早 已 发 生 了 改变 ， 无 论 是 Zend Framework 的 PDO 方式 、 
还 是 Symfony 的 Yml 表达 式 都 能 够 很 好 地 解决 直接 使 用 字符 串 市 来 的 安全 问题 ， 并 能 有 效 
提高 开发 效率 。ThinkPHP 作为 一 款 流 行 的 MVC 框架 ， 同 样 也 提供 了 多 种 方式 ， 和 党 用 的 有 数 
组 表达 式 如 表 7-2 Drot, 


表 7-2 数组 查询 支持 的 表达 式 


> 作 H 不 D 
EQ $data[“user”]=array(“eq”, “ceiba”’) 
NEQ $data[“id”]= array(“neq”, za 
GT $data[“id”]= array(“gt”, “20”) 
EGT $data[“id”]= array(“egt”, “10”) 
LT $data[“id”]= array(qdt “50”) 
ELT $data[“id”]= array(“It”, “100”) 
LIKE $data[ title”]= array(“like”, “ceiba%”) 
[NOT] BETWEEN $datafid] = array('between','1,8') 
[NOT] IN $datafid] = array('not in','1,5,8" 
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定义 好 表达 式 后 ， 束 可 以 直接 作为 参数 传递 给 where 连 员 操 作 了 ， 如 以 下 代码 所 示 。 
<?php 
class IndexAction extends Action { 
AAS 
public function index (){ 
ŞarticleO0bj=M ("Article"); 
$data["add_user"]=array ("eq", "ceiba"); 
$data["content"]=array ("like", "苹果 %")， 
$rows=$articleOb]j->where ($data)—>select () ; 
Şthis->assign("list",Ş$rows); 
Şthis->display ();}; 
| 
} 


?> 


如 上 述 代 但 所 示 ，where 传 入 的 参数 不 再 是 前 面 的 字符 串 ， 而 是 一 个 多 维 数组 ， 这 些 数 
组 元 素 最 后 被 转换 成 标准 的 SQL 语句 ， 如 以 下 代码 所 示 。 

SELECT * FROM "ek article" WHERE ( ‘add user" = 'ceiba' ) AND ( `content` LIKE '%¥ 
Rs' ) 

读者 可 以 使 用 echo $articleObj->getLastSql0; 得 到 转换 后 的 SQL 语句 。 上 述 代 人 码 中 共有 
两 个 查询 条 件 : add_user 和 content。 它 们 之 间 的 关系 为 AND (默认 情况 )， 即 两 个 条 件 都 必 
须 成 立 。 如 果 和 需要 更 改 僵 询 关 系 为 OR， 可 以 使 用 $data['_logic'] = 'OR' 数 组 元 双 值 进行 更 改 。 

从 上 述 代 但 中 可 以 看 出 ， 碍 询 表达 陈 可 以 完美 地 代 蔡 传统 的 字符 串 碍 询 表达 式 ， 并 市 来 
了 非常 好 的 效果 ， 不 会 降低 开发 效率 ， 建 议 读者 采用 这 种 方式 。 男 外 查询 表达 式 是 不 区 分 大 
四 与 | 的 

2. 普通 查询 

所 谓 的 普通 查询 是 相对 功能 而 言 的 ， 事 实 上 所 有 查询 语言 都 是 以 普通 查询 为 基础 的 ， 在 
实际 应 用 开 友 中 读者 没 必 要 区 分 查询 分 类 ， 因 为 无 论 哪 种 查询 都 是 连 叶 操作 的 结果 ， 不 同 的 
只 是 where 方法 中 的 参数 。 前 面 所 介绍 过 的 枉 询 语句 都 可 归 为 普通 查询 。 通 常情 况 下 ， 兽 通 
查询 都 会 返回 数据 集 ， 例 如 使 用 select 或 find 等 方法 。 由 于 普通 查询 在 前 面 的 内 容 中 已 经 多 
REIM, GI rm, 

3. 区 间 查 询 

区 间 查 询 就 是 需要 查询 的 结果 限制 于 两 个 或 两 个 以 上 的 查询 条 件 。 例 如 需要 查询 id>10 
并 且 id<30 之 间 的 数据 ， 这 束 限 制 了 结果 为 id 字段 11 一 29 之 间 的 数据 ， 如 果 使 用 传统 的 
SQL 语句 编号 ， 代 但 如 下 所 未 。 

Select * from tpk_article where id>10 and id<30 

如 果 使 用 查询 表达 式 ， 那 么 可 以 使 用 多 个 数组 进行 表达 。 此 时 的 id 数组 元 素 值 束 会 变 
为 一 个 多 维 数组 ， 如 以 下 代码 所 示 。 


<?php 


class IndexAction extends Action { 

E 

public function index (){ 
ŞarticleO0bj=M ("Article"); 
$data["id"]=array (array ("gt",5),array ("lt","8"),"and"); 
Şrows=$article0bj->where ($data)—>select  ();}; 
Şthis->assign("list",Ş$rows); 
Şthis->display ();}; 
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} 


如 上 述 代码 所 示 ， 数 组 id 元 素 的 值 使 用 了 一 个 多 维 数 组 ， 多 维 数 组 中 每 个 数组 元 素 代 
表 一 个 得 询 条 件 ， 最 后 一 个 数组 元 素 为 得 询 关 系 ， 可 选 的 值 有 and 或 者 or， 如 条 为 空 则 使 用 
and 默认 值 。 

4. 组 合 查 询 

组 合 查 询 可 以 对 复杂 的 多 个 查询 条 件 进行 封闭， 并且 能 够 在 数组 元 素 中 直接 使 用 学 符 
嘻 ， 最 终 的 结果 由 所 有 元 素 值 决定 。 比 如 需要 查询 id>5 或 者 tite GEA SGER” WEZ 
据 ， 最 后 还 限制 新 闻 来 源 地 区 为 “guangdons”， 奢 么 使 用 组 合 人 查询 代 人 码 就 变 得 人 简洁， 如 以 下 
代码 所 示 。 


<?php 
class IndexAction extends Action { 
EE 
public function index (){ 
ŞarticleO0bj=M ("Article"); 
// 条 件 1 
SCELEalvicw|=arrey ees 
saacal title] areav ( liket, Ts on 


Scaral Y odLewvl|s ome 

// 条 件 2 
$where["area"]=array (iech, "guangdong"); 
/ /条 件 拼接 


$Swhere["_complex"]=$where; 
Şrows=$article0bj->where (S$where)—>select () ; 
sthis->assign ("list",S$rows); 
sthas >Losolav( 


} 


2 


如 上 述 代码 所 示 ， 使 用 _complex 数组 元 素 对 2 个 查询 条 件 进 行 拼接 ， 如 果 有 更 多 的 条 件 
需要 拼接 ， 依 此 类 推 即 可 。 最 终 的 SQL 代码 如 下 所 示 。 

SELECT*FROM tpk article WHERE( area 'guangdong (AND arca si Ouangdongni? 

5. 统计 查询 

如 果 严 格 来 区 分 ， 统 计 但 询 并 不 算 查 询 语言 的 讲述 范畴 ， 因 为 在 ThinkPHP 中 要 进行 字 
段 数 值 统计 并 不 需要 额外 定义 数组 元 票 ， 只 十 要 更 改 显 示 方 式 即 可 。 例 如 普通 合 询 使 用 
select 进行 数据 显示 ， 而 统计 得 询 根据 统计 需求 有 各 目的 显示 方法 。 当 然 统计 得 询 是 可 以 结 
合 前 面 的 各 种 奏 询 方式 进行 使 用 的 ， 系 统 文 持 的 统计 奏 询 如 表 7-3 所 示 。 


SS 7-3 统计 查询 显示 方法 


Mo 说 H ZS 数 
Count 统计 指定 表 字 段 的 总 条 数 可 选 ， 参 数 为 字段 名 称 
Max 获取 指定 表 字 段 的 最 大 值 必 选 ， 参 数 为 字段 名 称 
Min 获取 指定 表 字 段 的 最 小 值 必 选 ， 参 数 为 字段 名 称 
Avg 获取 指定 表 字 段 的 平均 值 必 选 ， 参 数 为 字段 名 称 
Sum 获取 指定 表 字 段 的 合计 结果 必 选 ， 参 数 为 字段 名 称 
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统计 但 询 的 使 用 和 前 面 介 绍 的 连贯 操作 
下 代码 所 示 。 


<?php 
class IndexAction extends Action { 
ER 
public function index (){ 
ŞarticleO0bj=M ("Article"); 
$rows=S$articleObj->select () ; 
// 总 共 记录 数 


$num=$articleObj->count () ; 


ee 


似 ， 只 需要 更 改 最 后 一 个 显示 方法 即 可 ， 如 以 


Šthis->assign ("num", $num); 
Šthis->assign("list",Šrows); 
$this->display(); 


} 


?> 


7.2.4 ”使 用 原生 的 SQL 语言 


虽然 系统 已 经 将 数据 库 操作 进行 了 OOP 封装 ， 完 善 的 CURD 操作 及 表达 式 能 够 满足 大 
部 分 需求 ， 但 还 不 能 完全 代 蔡 传统 的 SQL 语言 ， 尤 其 在 一 些 复杂 的 SQL 查询 中 ， 所 以 系统 
也 提供 了 传统 的 SQL 语言 查询 。 在 Model 模型 中 ， 有 3 种 方式 可 以 使 用 原生 的 SQL 语言 ， 
下 面 分 别 进行 介绍 。 

1. query 方法 

query 方法 的 使 用 比较 稍 单 和 和 直观， 接受 的 参数 即 为 完整 的 SQL 语句 。 这 里 所 说 的 SQL 
语句 是 带 完 整数 据 表 名 的 《包含 前 绥 ) 查询 语 言 。guery 方法 返回 的 结果 是 一 个 数据 集 ， 可 
以 直接 分 配 到 模板 中 ， 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 


public function index (){ 
ŞuserOobj=M(); 
$sql="select * from tpk_user where id>0"; 
SOWweE Tn So ol 
Şthis->assign("list",Ş$rows); 
SE -display () 


} 


?> 


直接 使 用 query 方法 执行 SQL 语句 需要 给 出 完整 的 数据 表 名 ， 这 时 在 DB_PREFIX D 
置 文件 中 的 配置 项 束 变 得 室 无 作用 了 ， 如 条 数据 表 前 绥 一 旦 变更 ， 怠 会 给 代 但 维护 珊 来 严 
重 的 困难 。 这 里 可 以 使 用 两 种 方式 进行 解决 : 一 种 直接 用 使 用 C ep är DB_PREFIX 表 
前 级 ， 以 变量 的 方式 蕉 换 SQL 语句 中 的 数据 表 名 称 ; 另外 一 种 是 将 当前 动作 名 或 方法 名 改 
成 与 数据 表 名 相同 ， 在 SQL 语句 中 使 用 _TABLE_ 蔡 换 符 进行 蔡 换 。 使 用 原生 SQL 语句 
后 ， 所 有 的 参数 入 口 需 要 手动 过 滤 ， 营 用 的 过 小 函数 有 mysql_escape_string、addslashes、 
escapeshellarg 等 。 

2. execute 方法 

execute 方法 和 query 方法 是 一 样 的 ， 使 用 方式 也 一 样 ， 唯 一 不 同 的 是 execute 方法 不 会 
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返回 数据 集 ， 也 不 会 返回 受 影 啊 行 数 ， 甚 至 连 错误 信息 都 不 返回 。 通 笛 情 况 下 query 用 于 获 
取 数 据 ，execute 方法 用 于 后 从 插入 、 更 新 数 据 。 如 以 下 代 公 所 示 。 


<?php 
class IndexAction extends Action { 
public function index (){ 
ŞuserOobj=M(); 
$sql="insert into te user(user Dame, user em al ues /eae COM)"; 
ŞuserObj->execute ($sql); 
pehrs displayi; 
| 
| 


?> 


3. exp 操作 表达 式 

exp 操作 表达 式 是 介 于 CURD 表达 式 与 传统 SQL 之 间 的 一 种 查询 方式 ， 为 这 两 种 方式 
找到 了 一 个 平衡 点 。 在 exp 表达 式 中 ， 开 发 人 员 可 以 完全 使 用 标准 的 SQL 语句 (where 部 
T) 进行 数据 操作 ， 包 括 碍 询 、 更 新 、 删 除 等 ， 如 以 下 代码 所 示 。 


<?php 


class IndexAction extends Action { 
public function index (){ 
ŞarticleO0bj=M ("Article"); 
Saatal eitle" array tepi like eee and add user ~ oeiba T); 
$rows=$articleObj->where ($data)->select (); 
Strals=->assLeom El 
SE -display (); 


} 


p> 


原生 的 SQL 语句 建议 只 用 于 一 些 特殊 的 操作 ， 因 为 原生 的 SQL 语句 是 有 一 定 的 局 限 性 
的 ， 比 如 部 著 多 数据 库 应 用 时 ， 需 要 处 理 各 种 数据 库 SQL 语言 乙 间 的 过 民 。 此 外 ， 使 用 原 
生 的 SQL 得 询 需要 该 者 手动 进行 得 询 缓 仔 〈exp 表达 式 除 外 )。 


7.3 关联 模型 ( RelationModel ) 


关联 模型 是 一 种 解决 多 表 参 与 CURD 操作 的 数据 表 模 型 ， 它 以 简单 、 直 观 的 方式 将 原 
本 需要 多 个 步 又、 多 条 SQL 才能 完成 的 操作 进行 了 组 件 化 ， 对 于 开发 人 员 而 言 ， 几 乎 不 需 
要 任何 SQL 关联 查询 的 知识 ， 只 需要 定义 数组 元 素 即 可 。 


7.3.1 关联 关系 


在 学 习 关 联 模型 前 ， 首 先 需 要 理解 关联 模型 的 类 型 ， 只 有 理解 类 型 之 间 的 关系 ， 才 能 真 
正 理解 关联 模型 。 例 如 数据 库 里 有 文章 表 、 评 论 表 、 用 户 表 、 管 理 员 表 ， 假 设 以 评论 表 为 参 
照 表 ， 评 论 表 中 的 数据 每 条 都 对 应 一 篇 文章 。 对 于 评论 表 而 言 ， 评 论 表 与 文章 表 之 间 的 关系 
即 为 一 对 一 关系 ， 但 评论 表 为 主 表 ， 所 以 关系 描述 为 HAS_ONE. 

评论 内 容 又 必须 隶属 于 用 户 表 中 的 某 个 用 户 ， 那 么 评论 表 与 用 户 表 之 间 的 关系 属于 一 对 
一 关系 ， 但 以 用 户 表 为 主 ， 所 以 此 时 的 评论 表 与 用 户 表 之 间 的 关系 为 BELONGSTO (反之 
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依然 为 HAS_ONE)。 一 篇 文革 可 以 对 应 车 干 条 评论 数据 ， 此 时 文章 表 与 评论 表 之 间 的 关系 
为 一 对 多 ， 摘 述 为 HAS_MANY。 一 条 评论 可 以 被 多 个 管理 员 管 理 ， 同 时 多 个 管理 员 可 以 同 
时 管理 同一 条 评论 ， 此 时 评论 表 与 管理 员 表 之 间 的 关系 为 多 对 多 ， 描 述 为 MANN TO 
MANY. 
以 上 这 个 形象 的 例子 充分 地 说 明了 ThinkPHP 关联 模型 的 3 种 关系 5 种 分 类 ， 这 5 种 分 
类 系统 已 经 定义 为 了 5 个 常量 ， 如 表 7-4 所 示 。 


SS 7-4 关联 模型 关系 表 


分 类 常量 名 说 明 关 A 
HAS_ONE 一， 评论 表 与 文章 表 之 间 的 关系 
ONE_TO_ONE 
BELONGS_TO | 一， 用 户 表 与 评论 表 之 间 的 关系 
HAS MANY 才 多 ， 文 章 表 与 评论 表 之 间 的 关系 


ONE TO_MANY 


BELONGS_TO | 一， 评论 表 与 用 户 表 之 间 的 关系 
MANY TO MANY 才 多 ， 评 论 表 与 管理 员 表 之 间 的 关系 MANY TO MANY 


在 实际 应 用 开发 中 ， 最 常用 的 为 HAS. sing 和 HAS MANN 关系 ，BELONGS TO 适用 
于 ONE TO_ONE 及 ONE_TO_MANY。 通常 情况 下 ，BELONGS_TO 用 于 改变 HAS ONE 
和 HAS_MANY 的 关联 关系 参照 表 位 置 。 


7.3.2 ”关联 定义 

使 用 关联 模型 ， 重 点 在 于 数据 模型 关联 的 定义 ， 在 定义 模型 之 前 首先 需要 根据 表 7-3 的 
内 容 定义 好 数据 表 之 间 的 关系 。 定 义 关 联 模型 需要 在 目 定义 模型 中 定义 ， 并 且 需 要 确保 目 定 
义 模型 继承 于 RelationModel， 关 联 属 性 全 部 使 用 数组 元 素来 描述 。 

数组 元 素 根据 关系 类 型 的 不 同 而 有 所 不 同 ， 为 了 便于 排版 ， 分 别 使 用 序列 Al 代表 
HAS_ONE、A2 代表 BELONGS_ TO、A3 代表 HAS_MANY、A4 代表 BELONGS_TO, A5 
代表 MANY_TO_MANY， 如 表 7-5 所 示 。 


表 7-5 关联 模型 属性 定义 


数 组 项 SE 
mapping_type 关联 类 型 5 种 关系 类 型 常量 名 
class_name 关联 目标 数据 表 名 不 需 前 级 ) 中 
mapping_name SR | 7 
foreign key Eoc EE 
condition 领 外 关联 操作 条 件 | 
mapping po | 结果 集 旺 现 字段 E E EE 
e Die 字段 别名 [a A | 
parent_key Te y 
mapping limit EE aa TI i 
mappingorder | ame | 
lation foreign key | pagxa o e | | [7 
relation_table 多 对 多 的 中 间 关 联 表 名 称 am ill 
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如 表 7-4 所 示 ， 打 “v ”的 表示 该 数组 元 素 文 持 对 应 的 模型 类 型 ， 为 空 则 表示 不 文 持 。 
关联 模型 属性 定义 ， 需 要 在 $_link 目 定 义 成 员 属 性 中 定义 ， 如 以 下 代码 所 示 。 


<?php 
class ArticleModel extends RelationModel{ 


protected $_link=array ( 
// 关 联 模型 ( 表 ) 1 
'comment '=>array ( 
EIERE 


'class_name'=>'comment', 


y 
// 更 多 关联 模型 . . . 


) ;7 
/7 功能 代码 ... 


7.3.3 KEIRA AI CURD 


接 下 来 将 以 HAS_ ONE, BELONGS_TO, HAS_MANY 这 3 种 最 常见 的 关联 模型 类 型 作 
为 讲解 对 象 ， 深 入 浅 出 地 介绍 关联 模型 的 CURD 实际 应 用 。 

1. 数据 查询 

(1) HAS ONE 查询 

HAS_ONE 是 关联 模型 中 最 简单 的 一 种 查询 方式 ， 它 适用 于 ONE_TO_ONE ZS (HI: 
条 评论 对 应 一 入 文章)。 下 面 继续 以 前 面 创 建 的 tpk_article 表 作 


Wu EE 字段 类 型 
为 数据 表 ， 演 示 创 建 一 个 HAS_ONE 关联 模型 的 过 程 。 SE 
首先 创建 一 个 评论 表 ， 并 命名 为 tpk_comment， 评 论 表 有 5 ab int(11) 
个 字段 ， 其 中 ad 学 段 关 联 tpk_article 数据 表 中 的 id CR, Gr comment text 


comment time int{11)} 


就 简单 地 创建 了 HAS ONE 关联 模型 所 需要 的 条 件 ， 
tpk_comment 表 结 构 如 图 7-5 rz, 

然后 在 tpk_comment 表 插 入 儿 条 数据 ， 需 要 确保 ad 字段 中 ”图 7-5 tpk_comment 表 结 构 
的 值 与 tpk_article 表 中 的 id 值 相对 应 ，SQL 代码 如 下 所 示 。 


INSERT INTO tpk comment ( id , aid , comment , comment time , comment _user ) VALUES 

(CTT RE RE Ebert 

EE GA a; 

接 下 来 就 需要 使 用 目 定 义 模型 定义 tpk_comment 数据 表 与 tpk_article 数据 表 之 间 的 关联 
关系 了 。 打 开 自 定义 模型 ArticleModel， 并 让 模型 继承 于 RelationModel， 然 后 定义 成 员 属 性 


$_link， 代 码 如 下 所 示 。 


<?php 
class ArticleModel extends RelationModel{ 


回国 回 加 思 


Comment user int{11)} 


protected $_link=array ( 
// 关 联 模型 1 
'comment '=>array ( 
"mapping_type"=>HAS_ONE, 
"class_name "=>"comment", 
"mapping_name"=>"comment", 


UTOreign key =>WaLe, 
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"mapping_fields"=>array ("id","comment"), 
IR 
// 更 多 关联 模型 


} 


EK 


如 上 述 代 码 所 示 ，$_link 属性 值 是 一 个 多 维 关 联 数组 ， 其 中 comment 表示 需要 与 当前 模 
型 建立 关联 模型 的 名 称 《〈 即 不 市 前 缀 的 数据 表 名 )，$_link 属性 允许 定义 多 个 关联 模型 。 
Comment 的 值 为 一 个 关联 数组 ， 其 中 mapping type 元 素 指定 了 该 关联 模型 的 分 类 〈 人 参考 
K 7-3); 其 他 数组 元 聚 根据 类 型 不 同 而 有 所 区 别 (参考 表 7-4)， 这 里 除了 mapping_type 
外 ， 还 定义 了 class name., mapmping name, foreign key、mapping fields 元 素 ， 其 中 


foreign_key 是 必需 的 。 


LI 说明: foreign key 也 是 一 个 可 选项 ， 用 于 指定 目标 数据 表 与 当前 数据 表 进 行 关 联 的 有 效 字 段 ( 即 外 
键 ) ， 默 认 情 况 下 系统 会 采用 “数据 对 象 名 称 _id” 的 形式 为 该 元 素 赋 值 ， 例 如 关联 的 模型 为 user， 那 
么 foreign_key 的 值 为 会 被 设 为 user_id。 


定义 好 关联 属性 后 ， 接 下 来 就 可 以 在 动作 中 直接 使 用 了 。 调 用 目 定 义 关联 模型 需要 结合 
relation 连贯 方法 ， 款 认 情 况 下 relation 方法 是 关闭 的 ， 需 要 传 入 true 参数 ， 关 联 得 询 才 会 起 
作用 ， 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 


public function index (){ 
SarcielegoJ=D(VArcieleY) > 
Şrows=$article0bj->field('id,title')->relation (true)->select(); 
dump ($rows); 
} 
} 


?> 


关联 查询 可 以 使 用 所 有 Model 基础 模型 的 CURD 操作 方法 ， 例 如 find、delete 等 。 运 行 
效果 如 以 下 代码 所 示 。 


array(8) { 


E ve Eet, i 
EST, => eer, T1” 
["title"] => string(66) " 库 克 束 苹果 地 图 缺陷 向 用 户 道歉 : 未 提供 一 流体 验 " 
(nceommentnl => NULE 

} 

E EE 


[6] => array (3) { 


["id"] => string(1) "7" 
["title"] => string(12) "eeeeeeeeeeee" 
["comment"] => array (2) { 

["id"] => string(1) "1" 

["comment"] => string (15) "第 一 条 评论 " 
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} 


[7] => array (3) { 


["id"] => string(1) "8" 
["title"] => string(7) "fsdfsdf" 
["comment"] => array (2) { 


["id"] => string(1) "2" 


ek — 


["comment"] => string(15) "第 二 条 评论 " 


} 


如 上 述 代码 所 示 ， 由 于 由 于 评论 表 中 只 存在 ad 字段 为 7 和 8 的 数据 ，aid 是 tpk_article 
表 的 外 键 ， 所 以 只 有 id 为 7 和 8 的 文章 有 评论 数据 。 其 中 comment 即 为 关联 模型 中 的 内 
容 ， 默 认 情 况 下 返回 二 维 数组 ， 因 为 在 HAS_ONE 关系 中 ， 所 有 关联 模型 的 数据 都 只 存在 一 
条 ， 为 了 操作 方便 可 以 使 用 as_fields 属性 将 二 维 数组 结果 拆 分 为 一 个 单独 的 数组 元 素 。 打 开 


ArticleModel， 增 加 as_fields 元 素 ， 如 以 下 代码 所 示 。 


<?php 
class ArticleModel extends RelationModelt 
protected $_link=array ( 
了 全 


"Comment '=>array ( 


"mapping_type"=>HAS_ONE, 
"class_name "=>"comment", 
"mapping_name"=>"comment", 
EES DE Ee KA 
"mapping_fields"=>array ("id","comment"), 
"as_fields"=>"id:comment_id, comment", 

), 

/ /更 多 关联 模型 

); 
| 


如 上 述 代码 所 示 ， 由 于 合并 后 关联 模型 的 id 与 当前 模型 的 id 产生 冲突 ， 所 以 需要 使 用 
“:” 为 id 设 年 一 个 别名 “comment_id”。 定义 as_fields 元 素 后 ， 结 果 集 将 全 部 合并 为 一 个 关 
联 数 组 。 如 以 下 代码 所 示 。 


[6] => array (4) { 


["id"] => string(1) "7" 
["title"] => string(12) "eeeeeeeeeeee" 
["comment_id"] => string(1) "1" 
["comment"] => string(15) "第 一 条 评论 " 
} 
[7] => array (4) { 
["id"] => string(1) "8" 


["title"] => string(7) "fsdfsdf" 
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["comment_id"] => string(1) "2" 
[Tcomment"] => string(15) 第 二 条 评论 

} 

(2) BELONGS_TO 查询 

BELONGS_TO 是 适用 于 HAS ONE 和 HAS MANN 关联 关系 ， 这 里 继续 以 前 面 
HAS ONE 示例 为 例 ， 介 绍 BELONGS_TO 的 使 用 。 前 面 的 示例 中 ， 使 用 HAS ONE 演示 了 
一 条 评论 id 对 应 一 篇 文 草 ， 那 么 同样 一 条 评论 数据 只 对 应 一 个 用 户 id， 评 论 表 与 用 户 表 之 
间 的 关系 即 为 BELONGS_TO。 

这 样 一 来 上 只 需要 在 ComentModel 自 定义 模型 中 定义 user 表 为 关联 模型 ， 然 后 将 
tpk_comment 表 中 的 comment user 字段 与 tpk_user 表 中 的 id 字段 (默认 字段 ， 不 需 填写 ) 
建立 关联 关系 ， 那 么 就 实现 了 查询 tpk_comemnt 表 ， 同 时 得 到 ek user 表 数 据 中 的 用 户 数据 
( 即 得 到 评论 用 户 名 )。 

省 完 创建 目 定 义 模 型 ， 并 命名 为 CommentModel。 让 其 继承 于 RelationModel 模型 ， 然 
后 设置 关联 关系 为 BELONGS_TO， 如 以 下 代码 所 示 。 


<?php 
class CommentModel extends RelationModel{ 


protected $_link=array ( 

'user'=>array ( 
"mapping_type"=>BELONGS_TO, 
"class_name"=>"user", 
"mapping_name"=>"user", 


"foreign_key"=>"comment_user", 


} 


?> 


同样 可 以 使 用 as_fields 元 素 合 并 但 询 结 果 ， 这 样 在 调用 Comment 目 定 义 模型 时 ， 不 仅 
可 以 但 到 tpk_comment 表 数 据 ， 还 能 得 到 tpk_user 表 数 据 。 调 用 方式 与 HAS_ONE 方式 无 
异 ， 效 果 如 以 下 所 示 。 


array(2) { 


[0] => array(6) { 


["id"] => string(1) "1" 

["aid"] => string(1) "7" 

["comment"] => string (15) "第 一 条 评论 " 
["comment time"] => string(10) "1346401694" 
(comment user"] => string(1) "1" 


["user"] => array (3) { 
["id"] => string(1) "1" 
["user_name"] => string(9) "F" 


["user_email"] => string(12) "kf@86055.com" 
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(3) HAS_MANY 查询 

一 篇 文章 可 以 对 应 多 条 评论 ， 评 论 表 与 文章 表 之 间 的 关系 即 为 HAS_MANY 。 
HAS_MANY 是 关联 模型 中 最 常用 的 操作 ，HAS_MANY 的 使 用 非常 简单 ， 和 HAS ONE A 
比 ， 只 需要 修改 其 中 的 关系 类 型 即 可 ， 其 他 的 元 素 几 乎 不 用 修改 。 

当然 as _fields 元 素 只 适用 于 单条 数据 ， 而 HAS_MANY 是 用 于 查询 多 条 数据 的 ， 所 以 
as_fields 在 这 里 使 用 是 没 意 义 的 。 继 续 以 前 面 创建 的 HAS ONE 示例 为 例 ， 修 改 其 中 的 
mapping_type 类 型 后 ， 得 到 的 效果 正 是 一 条 新 闻 对 应 多 条 评论 数据 (前 提 是 评论 表 中 有 多 条 
aid 重复 值 的 评论 数据 )。 由 于 HAS_MANY 和 HAS_ONE 基本 相同 ， 在 此 就 不 做 演示 了 。 

至 此 ， 可 以 得 到 一 个 结论 ，HAS_MANY 与 HAS_ONE 是 一 对 取 反 的 关联 模型 ， 
HAS ONE 用 于 操作 单条 数据 ，HAS_MANY 用 于 操作 多 条 数据 。 

2. 数据 的 操作 

一 旦 定义 好 关联 模型 属性 ， 那 么 之 后 的 CURD 操作 都 不 需要 做 更 改 。 接 下 来 继续 以 最 
典型 的 HAS_MANY 关系 为 例 ， 介 绍 关 联 模型 的 数据 操作 。 关 联 模型 中 创建 数据 是 最 重要 的 
步骤 ， 也 是 确保 能 否 正 确 操 作 数 据 的 关键 ， 假 设 需要 为 tpk_article 表 添 加 一 篇 文章 数据 ， 同 
时 为 tpk_comment 表 增 加 两 条 评论 数据 ， 那 么 只 需要 在 add 数组 中 定义 comment 数组 元 素 ， 
然后 在 comment 中 赋值 就 等 于 为 tp_comment 赋值 了 ， 代 人 码 如 下 所 示 。 


<?php 


class IndexAction extends Action { 

public function index (){ 
SarcielegoJj=D(VArcieleY h? 
/7 Saral Vict ]=7; 
Şrows=$article0Obj->where ($data)->relation (true)->delete(); 
echo $articleO0Obj->getLastsql(); 
dump ($rows);*/ 
$data["title"]=" J Hpne"; 
$data["content"]=" 这 是 一 条 关于 体育 动态 的 新 闻 "; 
$data["category"]=" 体 育 新 闻 "， 
sdatal[l"adqd time"]=time (); 
$data ["comment"]=array ( 


array ("aid"=>9, "comment "=>" {k 6 JHT iE 1", "comment_time"=>time (),"comment_ 
user"=>1), 
array ("aid"=>9, "comment "=>" JE ES Sr H YF W ZP, comment time” =>time(), “comment ` 
user"=>1), 
); 
Şarticle0bj->relation (true)->add ($data); 
} 
} 


p> 


如 上 述 代码 所 示 ， 其 中 标 粗 的 即 为 关联 的 模型 ， 访 元 素 名称 需 要 与 ArticleModel AX 
模型 中 的 关联 模型 名 称 相 同 。 

既然 可 以 插入 数据 ， 那 么 更 新 数据 也 一 样 ， 上 只 需 增 加 where 条 件 限制 操作 并 将 add 操作 
改 为 save 操作 即 可 。 接 来 下 将 介绍 关联 模型 的 数据 删除 。 

关联 删除 是 关联 模型 中 最 典型 的 应 用 ， 也 是 最 单 用 的 一 种 功能 ， 许 多 大 型 的 关系 型 数据 
库 都 提供 这 种 功能 ， 在 传统 的 PHP 与 MySQL 开发 中 ， 要 达到 这 样 的 效果 ， 开 发 人 员 需 要 使 
用 join 或 者 视图 来 实现 ， 也 可 以 直接 在 PHP 代码 中 使 用 foreach 等 语句 来 实现 。 在 
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ThinkPHP 关联 模型 中 ， 要 删除 数据 变 得 非常 简单 ， 只 需要 将 关联 外 键 赋 参 给 delete 操作 方 
法 ， 即 可 实现 关联 删除 。 无 论 是 哪 种 关联 关系 ， 都 是 通用 的 ， 系 统 会 自动 删除 相关 联 的 数 
据 ， 如 以 下 代码 所 示 。 


<?php 


class IndexAction extends Action { 
public function index (){ 
$SarcLele0oJ=D ee est 
SE >relation (true) >deletce(9); 


} 


> 


如 果 只 需要 删除 tpk_comment 表 中 的 数据 〈 即 关联 模型 )， 那 么 只 需要 为 relation 操作 方 
法 传 入 comment 参数 即 可 。 


7.4 高 级 模型 ( AdvModel |) 


局 级 模型 是 在 基础 模型 的 基础 上 增加 了 多 项 强化 功能 的 数据 表 模 型 。 使 用 局 级 模型 除了 
能 够 实现 普通 模型 所 有 功能 外 ， 还 能 够 实现 数据 过 滤 、 操 作 限 制 、 延 到 操作 等 实用 功能 ， 此 
外 还 提供 了 儿 种 传 参 式 获取 数据 字段 的 实用 功能 。 局 级 模型 是 以 扩展 方式 整合 进 系统 的 ， 它 
用 于 普通 模型 操作 之 外 的 一 些 特殊 操作 ， 但 并 不 总 味 看 普通 模型 不 能 实现 品级 模型 的 功能 。 
本 节 将 介绍 高 级 模型 中 的 定位 查询 、 动 态 查询 、 内 容 存 文 本 等 实用 功能 。 


7.41 ”定位 查询 


定位 得 询 顾名思义 融 是 将 得 询 指针 定位 到 茶 条 记录 。 普 通 香 询 中 通 各 使 用 select 方法 获 
取 结 采集 ， 但 有 时 并 不 是 每 个 络 末 集 都 需要 获取 多 条 记录 的 ， 局 级 模型 对 select 进行 了 细 
化 ， 提 供 了 3 个 实用 方法 用 于 实现 定位 人 查询， 分 别 为 getN、first、last。 

(1) getN 

getN 方法 用 于 获取 符合 条 件 指定 位 置 的 记录 ， 授 受 的 参数 为 数值 类 型 ， 即 需要 获取 结果 
集中 指定 的 记录 人 位置， 参数 为 负数 时 则 获取 倒序 同 前 的 记录 ， 人 否则 即 正 序 同 后 的 记录 ， 如 以 
FLA zz, 

$User->where ('status =1')->order ('add_time')->getN (2); 

上 述 代 人 码 表示 获取 结果 集 第 3 条 记录 《〈 即 正 序 跳 过 2 条 )， 如 来 需要 获取 结果 集 的 倒 友 
第 2 条 ， 则 需要 传 入 负数 ， 如 以 下 代码 所 示 。 


$User->where ('status =1')->order ('add_time')->getN (-2); 

(2) first 

如 果 只 需要 获取 结果 集中 的 第 1 条 记录 ， 那 么 可 以 直接 使 用 快捷 方法 first， 该 方法 不 需 
要 赋 参 ， 代 码 如 下 所 示 。 


SUser->where (Istatus =1')->order ('add time')->first(); 


(3) last 
可 以 获取 正 序 第 1 条 记录 ， 当 然 也 可 以 获取 倒序 第 1 条 记录 ，last 用 于 获取 记录 集 的 最 
后 一 条 记录 ， 代 但 如 下 所 示 。 
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SUser->where (Istatus =1')->order ('add time')->last (); 


7.4.2 ”动态 查询 


如 有 果 只 是 获取 特定 的 字段 或 者 获取 指定 的 记录 ， 使 用 动态 会 询 更 加 局 效 。 动 态 合 询 免 除 
了 where 条 件 部 分 ， 直 接 动态 传 参 即 可 。 局 级 模型 中 一 共 提 供 了 3 个 操作 方法 用 于 实现 动态 
AW, TIÄ getBy、getFieldBy 和 top- 
(1) getBy 
getBy 方法 用 于 根据 茶 个 字段 的 值得 询 数据 ， 这 里 所 说 的 某 个 字段 是 不 确定 的 ， 是 一 个 
动态 的 字符 串 ， 例 如 getByEmail， 束 表示 根据 Email 字段 备 询 数据 ， 如 以 下 代码 所 示 。 
<?php 
public function index (){ 
ŞUser=new AdvModel ("User"); 
Şrows=$User->getByUserEmail("kfQ@86055.com"); 


dump ($rows); 


} 


} 


上 述 代 码 中 传 入 参数 “kf@86055.com ”， 即 表示 在 UserEmail 字段 中 查找 值 为 
kf@86055.com 的 记录 。 这 里 需要 注意 的 是 ， 在 tpk_user 表 中 并 不 存在 UserEmail 这 个 字段 ， 
但 结果 一 样 正 确 ， 这 是 因为 系统 默认 使 用 了 模型 与 数据 表 映 里 规划， 即 系 统 会 将 UserEmail 
这 个 字符 串 解 释 为 user email 字段 。 上 述 代码 的 转换 后 的 SQL 语句 如 下 所 示 。 

SELECT * FROM tpk user WHERE ( user email = 'kf@86055.com' ) LIMIT 1 

如 果 字 段 为 数字 类 型 ， 需 要 传 入 正确 的 数字 ， 例 如 getById(])。 

(2) getFieldBy 

getFieldBy 和 getBy 相关 似 ， 但 getFieldBy 不 仅 获取 到 茶 个 字段 指定 值 鸭 数据 ， 还 可 以 
根据 某 个 字段 值得 到 另 一 个 字段 的 值 。 假 设 根据 kf@86055.com 查询 结果 ， 得 到 该 用 户 的 
id， 那 么 可 以 动态 表示 为 getFieldByEmail(“kf@86055.com”, “id”)， 如 以 下 代码 所 示 。 

KE EE 


Ş$User=new AdvModel ("User"); 
Şrows=$User->getFieldByUserEmail ("kf@86055.com","id"); 


dump ($rows); 
} 
} 


?> 


上 述 代码 转换 后 的 SQL 语句 如 下 所 不 。 


SELECT ‘id FROM tpk user WHERE ( user email = 'kf@86055.com' ) LIMIT 1 

(3) top 

getN 操作 方法 用 于 跳 过 记录 ， 而 top 操作 方法 用 于 包含 记录 。top 操作 方法 是 一 个 动态 
方法 ， 不 需要 传递 参数 ， 它 的 表示 方式 直观 好 记 ， 例 如 需要 获取 前 $ 个 用 户 ， 那 么 就 表示 为 
top50。 严 格 意义 上 数字 5 并 不 是 一 个 合格 的 方法 命名 规则 ， 但 在 动态 操作 中 是 被 允许 的 。 

动态 查询 只 有 top 操作 方法 需要 使 用 AdvModel 高 级 模型 ， 其 他 两 个 并 不 需要 ， 这 里 为 
了 便于 讲解 ， 所 以 将 其 归 类 为 高 级 模型 。 需 要 说 明 的 是 AdvModel 模型 完全 适用 于 Model 基 
础 模型 ， 所 以 在 动态 模型 中 ， 基 础 模型 的 where、order 等 连贯 操作 都 是 通用 的 。 
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7.4.3 ”内 容 存 文本 


内 容 存 文本 是 一 项 非常 实用 的 功能 ， 将 数据 表 中 一 些 大 数据 量 的 了 字段， 如 txt KAF 
Et blob 类 型 字段 转 存 于 纯 文 本 文件 中 ， 能 够 有 效 地 减少 数据 表 的 容量 ， 提 高 数据 表 的 运行 
效率 ， 并 改善 数据 表 元 余 。 要 将 数字 表 字 段 改 为 纯 文 本 储存 方式 ， 只 需要 在 目 定义 模型 中 定 
义 $blobFields 成 员 属 性 值 即 可 ， 更 改 后 原先 的 CURD 操作 不 受 任 何 影响 ， 接 下 来 将 通过 一 个 
示例 演示 $blobFields 属性 的 使 用 。 

打开 ArticleModel.class.php 模型 文件 ， 将 ArticleModel 模型 的 父 类 改变 为 AdvModel， 
然后 定义 $blobField 成 员 属 性 值 ， 这 里 只 需要 改变 thk_article 数据 表 中 的 content 字段 储存 方 
式 为 纯 文 本 。 代 但 如 下 所 示 。 


<?php 
class ArticleModel extends AdvModel{ 
Protected SblobFields = array ('content'); 
i 
?> 


如 果 有 多 个 字段 需要 改 为 纯 文 本 储存 方式 ， 在 数组 中 添加 上 相应 的 字段 名 称 即 可 。 数 组 
中 定义 的 字段 可 以 是 一 个 独立 的 字段 ， 不 需要 数据 表 中 同时 存在 该 和 字段。 通过 前 面 的 步骤 ， 
接 下 来 的 CURD 操作 不 受 任何 影响 ， 如 以 下 代 人 码 所 示 。 


<?php 


class IndexAction extends Action { 
public function add(){ 

$data["title"] =" AAFAA"; 

odata KEE E EE 

şdata["area"]="guangdong"; 

sdatal"area"|]="ceiba"; 

şdata["add_time"]=time (); 

Sarticleobj = D('Article'!); 

if (ŞarticleO0bj->add(Ş$data)){ 
$this->success ("数据 插入 成 功 ")， 

}else { 
$this->error (" 数 据 插 入 失败 ") ; 

} 

ogc hes -display F 


} 


?> 


代码 执行 后 ， 储 存 的 文本 内 容 默 认 被 存放 于 home/Runtime/Data/Article/ 目 录 下 ， 文 件 以 
“主键 id_ 表 字段 ”的 方式 命名 ， 如 9_content.php。 

需要 注意 的 是 ， 字 段 内 容 转 存 文本 之 后 ， 相 应 的 数据 就 不 受 数 据 库 保 护 了 ， 系 统 将 以 IO 的 方 
却 取 出 文本 数据 ， 这 残 意 味 寿 内容 存 文本 的 字段 上 只 能 是 一 些 不 重要 或 者 不 需要 受 你 护 的 数据 。 


7.5 大 数据 支持 
在 这 个 信息 量 暴涨 的 时 代 ， 大 数据 存储 无 论 何 时 都 是 每 个 程序 员 必 须 重视 的 问题 。 在 小 
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网 站 中 ， 一 合 MySQL 服务 器 ， 单 个 数据 库 ， 甚 全 单个 数据 表 也 能 轻松 应 对 。 但 是 在 大 数据 
中 ， 这 种 染 构 无 疑 是 任命 的 。 假 设 有 1 亿 条 数据 记录 需要 实现 CURD AW, WREKE 
些 数 据 存 放 在 一 个 表 ， 那 么 再 强大 的 服务 器 也 承受 不 了 ， 这 时 就 可 以 使 用 数据 库 表 分 区 技术 
或 者 数据 库 分 库 技 术 来 将 1 亿 条 记录 分 开 存 储 ， 为 了 实现 但 询 和 号 入 互相 不 干扰 ， 还 可 以 使 
用 数据 库 谈 写 分 离 扩 术 实现 高 效 的 应 用 。 

这 里 所 说 的 大 数据 支持 是 指 传统 的 关系 型 数据 库 应 用 。 为 了 便于 讲解 ， 这 里 以 最 经 典 的 
MySQL 作为 示例 。MySQL 5.2 以 后 的 版 本 能 够 完美 地 支持 主 从 同步 、 表 分 区 、 数 据 均 衡 等 
只 有 商业 数据 库 才 支持 的 功能 ， 这 些 功能 是 确保 能 够 适合 大 数据 操作 的 关键，ThinkPHP 能 
够 对 这 些 功能 提供 民 好 的 应 用 开发 广 持 ， 比 如 多 数据 库 切换 、 数 据 表 分 区 、 读 写 分 离 等 。 


7.5.1 分 布 式 数据 库 


分 布 式 数据 库 并 不 是 指 多 个 数据 库 ， 严 格 意义 来 说 是 指 分 布 式 数据 库 服务 器 ， 也 称 服务 
句 集 群 。 比 如 MySQL 服务 器 集群 ， 只 是 由 于 习惯 性 的 问题 ， 多 数 程序 员 都 称 为 分 布 式 数据 
库 。 分 布 式 数据 库 的 特点 是 多 人 台数 据 库 服 务 器 轮流 对 外 提供 服务 ， 并 提供 元 余 、 容 灾 等 基本 
功能 ， 人 确保 操作 不 中 断 。 衡 量 一 个 MVC 框架 能 不 能 进行 分 布 式 数 据 库 开发 ， 首 先 需 要 看 
MVC 中 的 连接 对 象 能 否 智 能 并 顺利 地 切换 数据 库 服 务 占 。 默 认 情 况 下 ， 一 个 连接 对 象 只 对 
一 台数 据 库 服务 器 生效 。ThinkPHP 提供 完善 的 数据 库 连 接 驱 动 ， 并 且 能 够 同时 文 持 多 种 类 
型 数据 库 的 分 布 式 开发 。 默 认 情 况 下 提供 了 MYySQL 分布 式 数据 库 开 友 驱 动 ， 如 果 读 者 使 用 
的 是 其 他 数据 库 ， 需 要 上 自行 到 http://wwwi.thinkphp.cn/extend/driver.html 网 址 下 载 对 应 的 驱 
动 。 下 面 以 默认 的 MySQL 驱动 为 例 ， 详 细 介 绍 ThinkPHP 连接 分 布 式 数据 库 的 过 程 。 

1. 配置 数据 库 

ThinkPHP 人 秒 化 了 连接 分 布 式 数据 库 的 步骤 ， 开 发 人 员 不 需要 手动 创建 连接 对 象 和 释放 
连接 对 象 ， 只 需要 在 配置 文件 中 增加 数据 库 配 置 即 可 。 这 里 假设 在 原 有 的 MySQL 数据 库 服 
务 右 的 基础 上 再 添加 一 合用 于 存放 BBS 论坛 数据 的 MySQL DR ZS, ip 地 址 为 
192.168.1.10。 配 置 文件 如 以 下 代码 所 未 。 


<?php 


return array 人 


// "配置 项 "=>" 配 置 值 ' 


(pp TP => 'mysql', // 数据 库 类 型 

IDB HOST => 'localhost', // 服务 器 地 址 

'DB_NAME' => 'tp', // 数据 库 名 

DB U CER! => 'root', // 用 户 名 

'DB_PWD' => 'root', // 密码 

(pp EREPI X] => 'tpk_', // 数据 库 表 前 级 

EMPL L DELIN? Ee 

'TMPL_R_DELIM' a Eet 

'LAYOUT_ON'=>true, 

"DB_Con1"=>array ( 
'db_type' => mysal", // 数据 库 类 型 
'db_host' => '192.168.1.10', // 服务 器 地 址 
'db_name' => os", // 数据 库 名 
'db_user' => "roO", // RZ 
'db_pwd' => (OGEV, // ZWB 
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// . .更 多 数据 库 配 置 
de 


R 


如 上 述 代码 所 示 ， 新 添加 的 数据 库 配 置 项 命名 为 DB_con1， 该 名 称 是 自 定 义 的 ， 切 换 数 
据 库 时 需要 使 用 到 。 如 果 需 要 添加 更 多 集群 数据 库 服务 器 ， 只 需要 按照 格式 填写 即 可 。 配 置 
言 息 和 连接 方式 根据 驱动 情况 而 有 所 不 同 ， 比 例 使 用 DNS 连接 方式 ， 那 么 就 必须 按照 DNS 
的 方式 配置 。 

这 里 需要 注意 的 是 ， 新 增 的 数据 库 配 置 项 名 称 必须 为 全 小 写 ， 否 则 系统 将 不 能 正确 识别 
(以 ThinkPHP 3.0 为 例 )。 男 外 ， 新 增加 的 数据 库 配 置 不 支持 配置 数据 表 前 级 。 配 置 完成 后 ， 
原先 的 数据 库 配 置 会 被 系统 设 为 默认 数据 库 ， 并 分 配 编号 0; 而 新 增 的 DB_Conl 数据 库 分 
配 的 编号 为 1， 需要 开发 人 员 手 动 切换 。 

2. 使 用 数据 库 

分 布 式 数据 库 配 置 完成 后 ， 接 下 来 就 可 以 直接 使 用 了 。 假 设 bbs 数据 库 中 有 一 个 数据 表 ， 
并 命名 为 comm_bbs， 那 么 动态 切换 到 该 数据 表 是 非常 简单 的 ， 代 人 码 如 下 所 示 。 


<?php 


class IndexAction extends Action { 
public function index (){ 
suser = MII: 
$data = $user->db (1, 'DB_Con1')->table ("comm _bbs")->where ("classid =1")—>select (); 
echo $user->getLastsql(); 
dump ($data) ; 


} 


table 方法 是 一 个 重要 的 方法 ， 该 方法 用 于 指定 数据 表 。 前 面 已 经 讲述 过 新 增加 的 数据 库 
配置 不 支持 配置 表 前 级 ， 所 以 table 方法 必须 传 入 带 表 前 组 的 数据 表 名 称 。 但 是 如 果 新 增加 
的 数据 库 表 前 级 本 来 就 和 默认 数据 库 的 表 前 缀 是 一 样 的 ， 那 么 table 方法 是 可 以 省 略 的 。 如 
以 下 代码 所 示 。 


<?php 


class IndexAction extends Action { 
public function index (){ 
Şuser = M("Bbs"); 
$data = $user->db (1, 'DB_Con1')->where ("classid =1")->select(); 
echo $user->getLastsql(); 
dump ($data) ; 


} 


事实 上 分 布 式 数据 库 多 数 都 用 在 主 从 读 写 分 离 上 的 ， 这 也 就 意味 着 所 有 数据 库 的 表 结构 
都 相同 ， 这 种 情况 下 束 不 需要 table 方法 。 针 对 数据 库 读 写 分 离 ，ThinkPHP 还 专门 提供 了 更 
加 便捷 的 方式 。 
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7.5.2 Z594 


表面 介绍 过 多 数据 库 动 态 切 换 可 以 实现 数据 库 谈 写 分 离 ， 但 这 种 方式 是 需要 手动 切换 
的 ， 也 就 是 说 谈 的 时 候 需 要 开发 人 员 指定 数据 库 ， 写 的 时 候 也 是 如 此 。 事 实 上 ，ThinkPHP 
己 经 完美 地 解决 了 读 写 分 离 功 能 ， 本 广 将 会 详细 介绍 。 

读 写 分 离 的 本 质 是 数据 主 从 同步 ， 所 以 要 想 真 正 实现 读 写 分 离 ， 还 必须 要 在 数据 库 系统 
内 首先 实现 主 从 同步 复制 。 由 于 主 从 复制 在 每 种 数据 库 上 实现 的 方式 都 不 尽 相 同 ， 而 且 涉 及 
数据 库 系统 知识 ， 这 不 是 本 书 介绍 对 象 ， 所 以 接 来 下 不 会 深入 介绍 数据 库 主 从 复制 的 也 理 ， 
只 简单 介绍 实现 一 个 简单 的 数据 库 主 从 复制 过 程 。 

1. 配置 数据 库 主 从 复制 

仍然 以 MySQL 为 例 ， 介 绍 MySQL 主 从 复制 的 配置 过 程 。 要 实现 MySQL 主 从 复制 ， 
需要 确 你 主 MySQL 服务 右 的 版 本 写 大 于 或 等 于 从 MySQL 服务 右 的 版 本 号 ， 并 且 需 要 确保 
版 本 号 大 于 $.1。 除 此 之 外 还 需要 确保 开局 了 数据 库 bin-log 二 进 制 日 志和 慢 码 询 日 志 。 前 者 
是 必需 的 ， 而 后 者 是 可 选 的 ， 用 于 数据 库 维护 和 调 优 。 

由 于 bin-log 日 志 是 主 从 复制 的 关键 ， 所 以 这 里 有 必要 进行 简单 介绍 〈 更 多 资料 可 以 参 
Il MySQL 官方 手册 或 者 相关 书籍 )。bin-log 是 MySQL 数据 库 中 非常 有 用 的 一 种 高 级 日 志 ， 
它 完 全 地 记录 了 数据 库 系 统 对 数据 库 所 有 的 增 、 删 、 改 、 奏 行为 ， 但 不 会 记录 服务 器 本 喘 的 
健康 状态 。 假 设 程序 执行 了 select * from tpk_user where id=5 6J, ARA bin-log 将 会 完整 地 
记录 ， 并 且 为 该 操作 添加 上 position 18 - 

所 有 bin-log 日 六 默 认 存 放 于 数据 库存 放 目 录 里 《默认 为 data 或 var), bin-log 日 志 是 一 
种 二 进 制 文件 ， 使 用 一 般 的 文本 编辑 器 无 法 打开 〈 如 记事 本 、vi 等 )。MySQL 提供 了 
MySQLbinlog 管理 工具 用 于 查看 bin-log 日 志 。 利 用 bin-log 存放 完整 SQL 语句 的 特性 ， 系 统 
管理 人 员 可 以 轻易 地 从 bin-log 日 志 中 恢复 数据 。 同 样 的 原理 ，MYySQL 主 从 复制 并 不 是 真正 
地 复制 数据 表 中 的 信息 ， 而 是 复制 主 数 据 库 中 的 bin-log 日 志 ， 然 后 再 从 这 些 日 志 中 恢复 数 
据 ， 实 现 数据 的 复制 。 所 以 要 实现 主 从 复制 首要 前 提 残 是 开局 bin-log 日 志 ， 并 且 确 保 主 从 
数据 库 之 间 的 master bin-log 日 志 中 的 position 值 相同 。 

下 面 将 使 用 两 台 MySQL 数据 库 服 务 器 为 例 ， 简 单 介绍 主 从 复制 的 过 程 ， 满 足 本 章 后 续 
的 学 习 需 要 。 本 例 中 主 数据 库 的 ip 地 址 为 192.168.2.1; 从 数据 库 的 ip 地 址 为 192.168.2.2, 
读者 可 以 使 用 虚拟 机 来 模拟 上 述 环境 。 下 面 首先 配置 主 数据 库 。 

(1) 配置 主 MySQL 服务 器 

表面 多 次 提 到 要 实现 主 从 复制 ， 首 先 需 要 开局 bin-log 日 志 功 能 ， 无 论 主 服 务 器 还 是 从 
服务 器 都 一 样 。 首 先 查 看 主 服务 器 的 bin-log 日 志 状态 。 


mysql> show variables like 'log_bin'; 
十 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 十 一 一 一 一 一 一 一 十 


( variable name 1 value 


十 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 十 一 一 一 一 一 一 一 十 
oom | ON | 
十 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 十 一 一 一 一 一 一 一 十 


1 row in set (0.00 sec) 


如 果 结 末 为 ON 则 表示 bin-log 日 六 已 经 处 于 激活 状态 ， 人 否则 需要 打开 MySQL 配置 文 
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48 # Replication Master Server (default) 

49 # binary logging is required for replication 

DU log-bin=mysql-bin 

作为 主 服务 器 ， 还 需要 配置 服务 器 的 server-id， 该 值 是 唯一 性 的 ， 是 主 从 服务 的 重要 标 
识 ， 通 常情 况 下 主 服务 器 的 serverid 设置 为 1， 如 以 下 代码 所 示 。 

55 # required unique id between 1 and 2^32 - 1 

56 # defaults to 1 if master-host is not set 

57 # but will not function as a master if omitted 


58 server-id = 


配置 完成 后 ， 保 存 配置 文件 ， 重 局 MySQL 数据库。 至 此 ， 主 服务 器 就 基本 配置 完成 了 ， 
剩 下 的 只 需要 配置 一 个 专用 于 同步 数据 的 用 户 即 可 。 为 了 安全 ， 在 此 只 赋 给 同步 用 户 复制 数据 
的 权限 ， 旋 者 可 以 使 用 可 视 化 的 phpmyadmin 来 议 置 ， 也 可 以 使 用 命令 行 操作 ， 命 令 如 下 代 侣 
frz, 


>mysql> grant replication slave on *.* to SyncaQl92.168.0.210 identified by 


'syncQadmin'; 


>mysql> flush privileges; 


如 果 使 用 phpmyadmin 管理 工具 ， 创 建 的 用 户 只 需要 确保 REPLICATION SLAVE 被 打 
“v” 即 可 ， 如 图 7-6 frz, 


管理 


GRANT 

SUPER 

PROCESS 

RELOAD 
SHUTDOWN 

SHOW DATABASES 
LOCK TABLES 
REFERENCES 
REPLICATION CUENT 
REPLICATION SLAVE 
CREATE USER 


< 


图 7-6 选择 REPLICATION SLAVE 


ASR, ERAAN BOTKE tp 中 的 表 和 数据 导出 ， 以 便 
导入 到 从 服务 器 ， 确 保 主 从 数据 库 中 的 数据 一 致 。 读 者 可 以 使 用 mysqldump 导出 ， 也 可 以 使 
用 phpmyadmin 导出 。mysqldump 命令 如 下 。 

mysqldump -uroot proot eP F - /tmp/tp-sgl; 


以 上 是 Linux 的 导出 路 径 ， 如 果 是 Windows 系统 ， 需 要 做 对 应 的 更 改 。mysqldump 工具 
位 于 MySQL 安 儿 路 径 的 bin 目录 (例如 /usr/local/mysql/bin)。 

(2) 配置 从 MySQL 服务 器 

站 先 创 建 一 个 数据 库 用 于 存放 主 数据 库 中 的 数据 ， 这 里 将 从 服务 器 的 数据 库 命 名 为 tp 
(5.1.7 后 的 厂 本 需要 确保 主 从 一 致 )， 然 后 将 前 面 导 出 的 tp.sql 数据 导入 到 该 数据 库 中 ， 接 看 
将 印 .sql 文件 复制 到 从 服务 左上 ， 最 后 执行 导入 步骤 ， 如 以 下 代码 所 示 。 


>mysql -uroot -proot tp2 -v -f</tmp/tp.sql 


读者 也 可 以 使 用 可 视 化 的 phpmyadmin 导入 数据 。 但 如 果 使 用 phpmyadmin 导入 时 ， 主 
从 数据 库 都 需要 使 用 reset master 命令 重 置 bin-log 日 忘 。 
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导入 数据 之 后 ， 根 据 前 面 讲述 的 方式 查看 是 否 已 经 启动 bin-log 日 志 。 如 果 没有 ， 需 


打开 配置 文件 开局 bin-log 日 志 功 能 ， 然 后 设置 配置 文件 中 的 server-id 为 2， 如 以 下 代码 所 
不 。 


[mysqld] 

server-id=2 

log-bin=mysql-bin 

replicate-do-db=tp # 需 要 同步 的 数据 库 

replicate-ignore-db=mysql # 不 需要 同步 的 数据 库 

replicate-ignore-db=test # 不 需要 同步 的 数据 库 

修改 配置 文件 后 ， 需 要 重新 启动 MySQL， 以 便 让 新 配置 生效 。 最 后 进入 MySQL 命令 
行 工具 ， 执 行 同步 操作 ， 如 以 下 代码 所 示 。 


mysql> stop slave; 
Query OK, 
mysql> change master to 

-> master_host='192.168. 


-> master_user='sync', 


O rows affected (0.01 sec) 


E 


-> master_password='sync@ladmin'; 


Query OK, 


mysql> start slave; 


现在 可 以 使 用 show slave 


O rows affected (0.01 sec) 


status 命令 但 看 同步 状态 ， 如 果 Slave _IO_Running 和 


Slave_SQL_Running 这 两 项 都 为 yes 则 表示 同步 成 功 ， 如 以 下 代码 所 未 。 


mysql> show slave status \G 
大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 
Slave_IO State: 
Master_Host: 

Master_User: 

Master_Port: 

E LE SEA 
Master Log biller: 


Read Master Log Dos: 


Relay_Log_File: 
Relay_Log_Pos: 


Relay_Master_Log_File: 


Slave_IO_Running: 
Slave_SOL Running: 
Replicate Do_DB: 


Replicate Ignore_DB: 


Replicate Do Table: 


d 
Waiting for master to send even 
LIR a MHI Aa dl 
sync 
3306 
60 
mysql-bin.000001 
106 
mysql-relay-bin.000002 
252 
mysql-bin.000001 
Yves 


row 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 大 


Yes 
tp 
mysql,test 


Replicate_Ignore_Table: 
Replicate_Wild_Do_Table: 
Replicate_Wild_Ignore_Table: 


Last_Errno: 
Last_Error: 
Skip_Counter: 
Exec_Master_Log_Pos: 
Relay_Log_Space: 
Until_ Condition: 
Until Log_File: 
Until_ Log_Pos: 
Master_SSL Allowed: 
Master_SSL CA File: 
Master_SSL CA Path: 


O 
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Master_SSL_Cert: 
Master_SSL_Cipher: 
Master SSL Keys 
Seconds_Behind_Master: 0 
Master_SSL_Verify_Server_Cert: No 
Last_IO Errno: 0 
Lastr LO EE 
Last_SQOL Errno: 0 
Last_ SOL Error: 
Replicate Ignore Server Ida: 
Master_Server_Id: 1 
1 row in set (0.00 sec) 


该 者 可 以 在 主 数据 库 中 对 tpk_user 数据 表 进行 增删 、 改 、 会 等 操作 ， 从 数据 库 
(192.168.2.10〉 中 的 tpk_user 数据 表 同 样 也 会 跟着 发 生 改变 。 接 下 来 将 结合 ThinkPHP 实现 
数据 的 读 写 分 离 。 


D 提示 : 在 5.1.7 以 前 的 版 本 中 ， 从 服务 器 的 同步 配置 是 需要 在 MySQL 配置 文件 中 进行 配置 的 。 但 新 版 
只 需要 在 配置 文件 中 配置 server-id 和 指定 需要 同步 的 表 及 不 需要 同步 的 表 ， 然 后 使 用 命令 执行 同步 操作 
即 可 ， 这 点 与 老 版 本 区 别 较 大 ， 需 要 读者 注意 。 


2. 实现 读 写 分 离 

前 面 只 是 配置 了 数据 库 主 从 同步 功能 ， 主 从 同步 通常 用 于 数据 库 读 写 分 离 。ThinkPHP 
提供 了 完善 的 恋 写 分 离 功能 ， 开 发 人 员 不 需要 手动 切换 数据 库 。 什 么 时 候 读 ， 什 么 时 候 写 系 
统 会 自动 判断 。 读 数据 时 系统 会 操作 从 服务 器 ， 而 写 数据 时 系统 会 操作 主 服务 器 。 最 终 由 数 
据 库 实现 同步 ， 这 束 是 一 个 最 典型 的 数据 库 读 写 分 离 ， 下 以 将 以 前 面 配置 好 的 两 台 主 从 数据 
库 为 例 ， 详 细 介 绍 实现 读 写 分 离 。 

首先 打开 项 目下 的 数据 库 配置 文件 ， 修 改 其 中 的 数据 库 连 接 参 数 。 要 实现 多 数据 库 连 
接 ， 只 需要 使 用 “,” 分 隔 多 人 台 服 务 器 即 可 ， 如 以 下 代码 所 示 。 

<?php 

return array( 


//' MEAW =>' 配置 值 ' 
URE CASE INSENSITIVE] true, 


"DB_DEPLOY_TYPE"=>1, // 是 否 启用 分 布 式 
'DB_RW_SEPARATE' =>true, / /是否 启 用 智能 读 写 分 离 
(pp TYPES => 'mysql', // 数据 库 类 型 
Ne Ooi 三 
'DB_NAME' => 'tp', // 数据 库 名 

(pp USER! => 'root,root', // 用 户 名 
'DB_PWD' => 'root,root', // 密码 
(pp PRET => 'tpk_', // 数据 库 表 前 级 
"project name HIR ÉIS rn, 

“TMEL E DELTIM! => "<!l=={", 

UTME LIR DELIMI z eE 


'LAYOUT_ON'=>true, 
JI: 
如 上 述 代 码 所 示 ， 要 局 用 读 写 分 离 只 需要 设置 DB BW SEDARATE X true 即 可 。 然 后 
其 他 的 配置 信息 和 普通 的 配置 信息 相差 不 大 。DB_HOST 用 于 配置 服务 器 ip， 多 个 服务 器 使 
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用 “,” 隔 开 ， 排 在 第 1 个 位 置 的 表示 主 服务 器 〈 即 写 入 服务 器 )， 排 在 后 面 的 默认 都 会 被 分 
配 为 从 服务 器 《〈 访 服务器); DB_USER 用 于 配置 服务 登录 用 户 ， 顺 序 与 DB_HOST 相对 应 ， 
如 果 分 布 式 数据 库 所 有 登录 用 户 都 相同 ， 可 以 只 输入 一 个 登录 名 即 可 ; DB_PWD 与 
DB_HOST 配置 过 程 一 样 ， 在 此 不 再 细 述 。 

配置 文件 配置 好 后 ， 现 在 就 可 以 在 动作 中 测试 读 写 分 离 了 ， 如 以 下 代码 所 示 。 

Ee IndexAction extends Action { 


// Æw 


public function index (){ 
ŞarticleO0bj=M ("Article"); 
$rows=S$articleObj->select () ; 


dump ($rows); 

} 

IA 

public function add(){ 
SarticleübizchM(iArticeleini: 


$datal"title"]=" 读 写 分 离 测 试 "， 
$data["add_user"]="ceiba"; 
şdata["area"]="shanghai"; 
$data["category"]=" 教 育 新 闻 "; 
$data["content"]=" 读 写 分 离 测试 --- 内 容 "; 


if (SarticleObj->add ($data)) { 
Sthis->success ("数据 添加 成 功 ")， 
}else { 
$this->error ("数据 添加 失败 ") ; 
| 
} 
} 


Se 动作 执行 后 ， 读 者 可 以 首先 查看 192.168.2.1 主 服务 器 ， 然 后 再 观察 192.168.2.10 从 
服务 器 ， 可 以 看 到 这 两 台 服 务 噩 的 数据 是 同步 更 新 的 。 通 过 读 写 分 离 ， 能 够 有 效 地 提高 数据 
库 的 负载 能 力 。 


7.5.3 数据 表 分 区 


使 用 数据 库 读 写 分 离 能 够 有 效 地 解决 流量 负载 ， 提 高 连接 并 发 ， 是 大 型 网 站 常用 
的 技术 。 但 只 有 读 写 分 离 还 不 能 解决 前 面 提 到 过 的 1 亿 条 数据 存储 的 问题 。 通 常情 况 
下 ， 管 理 员 会 将 这 1 亿 条 数据 进行 水 平分 库 ， 但 这 对 于 开发 人 员 来 说 需要 面临 众多 的 
问题 ， 尤 其 是 过 多 切换 数据 库 将 降低 开发 效率 和 运行 效率 ， 并 且 其 后 期 程序 扩展 及 维 
护 是 非常 困难 的 。 

商业 数据 库 通 常会 使 用 数据 表 水 平分 区 功能 来 处 理 超大 型 数据 。MySQL 5.1 以 上 版 本 
《社区 版 ) 通过 插件 的 方式 能 够 完美 支持 表 分 区 功能 ， 便 得 MySQL 的 储存 性 能 得 到 了 质 的 
提升 。ThinkPHP 本 和 喘 也 参照 了 MySQL 表 分 区 功能 ， 提 供 了 简单 的 数据 库 分 表 功 能 。 事 实 
上 ， 所 谓 的 数据 库 分 表 是 在 开发 人 员 手 动 创建 数据 表 的 基础 上 ， 然 后 根据 设 定 的 规则 ， 分 步 
对 各 表 进 行 操 作 ， 实 现 应 用 层面 上 的 简单 分 表 功 能 。 由 于 处 理 逻 辑 在 程序 内 实现 ， 这 对 数据 
库 的 后 期 维护 无 疑 是 困难 的 ， 所 以 本 节 将 介绍 MySQL 中 的 表 分 区 功能 ， 人 代替 ThinkPHP 中 
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的 数据 分 表 功 能 。 

1. 数据 表 分 区 概述 

在 MyISAM 引擎 中 ， 传 统 的 数据 库 文件 1 个 表 对 应 3 个 文件 。 以 tpk_user 为 例 ， 用 于 
存放 表 结 构 的 tpk user frm, 用 于 存放 数据 的 tpk_userMYD; 以 及 用 于 存放 索引 文件 的 
tpk_user.MYI。 随 着 数据 的 快速 增长 ，tpk_userMYD 和 tpk_user.MYI 文件 大 小 也 会 激增 ， 
如 采 单 个 文件 超过 4GB， 那 么 数据 库 的 IO 操作 将 会 明显 变 慢 ， 频 索 的 操作 将 会 使 数据 库 
系统 谈 写 变 得 困难 ， 这 时 残 需要 将 数据 表 文 件 进行 切 制 ， 限 制 单 个 件 在 可 探 的 大 小 之 内 。 
数据 表 分 区 技术 残 是 用 于 切割 数据 表 文 件 的 一 项 实用 技术 ， 从 MySQL 5.1 开始 以 插件 的 方 
式 提 供 ， 要 和 谷 看 数据 库 系 统 是 否 文 持 数 据 表 分 区 功能 ， 可 以 使 用 show plugin 命令 但 看 ， 


如 以 下 代 但 所 示 。 
>mysql> show plugins; 
二 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 十 一 一 一 一 一 一 一 一 一 一 十 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 十 一 一 一 一 一 一 一 一 一 十 一 一 一 一 一 一 一 一 一 十 
| PARTITION | ACTIVE | STORAGE ENGINE ENOCE | GPL | 
十 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 二 一 一 一 一 一 一 一 一 一 一 十 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 十 一 一 一 一 一 一 一 一 一 十 一 一 一 一 一 一 一 一 一 十 


DU (DD PARTITION 插件 ， 那 么 资 明 当前 数据 库 系 统 文 持 表 分 区 功能 ， 人 否则 需要 在 
configure 时 加 入 相应 的 选项 (--with-plugins=all --with-PARTITION )。 如 果 是 Windows 下 的 
MySQL， 默 认 已 经 开启 表 分 区 功能 。 

2. 表 分 区 分 类 

在 确保 数据 库 系统 文 持 表 分 区 后 ， 就 可 以 创建 表 分 区 了 。MYySQL 共 文 持 4 种 分 区 类 型 ， 分 
别 为 RANGE、LIST、HASH、KEY。 在 生产 环境 下 ， 最 第 用 的 是 RANGE 分 区 和 LIST 分区。 

(1) RANGE 分 区 

RANGE 分 区 是 根据 分 区 规则 中 设 定 的 学 段 值 来 分 区 的 ， 设 定 的 字段 值 必须 是 能 够 有 序 
变化 并 唯一 的 数值 ， 例 如 时 间 、id 等 。 RANGE 的 分 区 规则 不 需要 开始 序列 ， 只 需要 结尾 序 
列 即 可 。 如 以 下 代码 所 示 。 


CREATE TABLE IF NOT EXISTS ‘tpk user ( 
`id`ò int (11) NOT NULL AUTO_INCREMENT, 
`user_name` char (20) NOT NULL, 
`user_email` char(40) NOT NULL, 

"elass icl nme (LL) NOT NU 
PRIMARY KEY (`id`), 
KEY "user name (`user_name`,` user email ) 

) ENGINE=MyISAM DEFAULT CHARSET=utf8 

PARTITION BY RANGE (id) ( 

PARTITION p0 VALUES LESS THAN (3), 
PARTITION p1 VALUES LESS THAN (6), 
PARTITION p2 VALUES LESS THAN (9), 
PARTITION p3 VALUES LESS THAN (12), 
PARTITION p4 VALUES LESS THAN MAXVALUE 


1: 

如 上 述 代 码 所 示 ， 表 示 使 用 RANGE 分 区 ， 并 且 以 id 作为 分 区 依据 。 当 id 到 达 3 时 则 
创建 p0 分 区 ， 依 次 类 推 ， 当 id 超过 12 时 将 存放 于 p4 分 区 中 。 

(2) LIST 分 区 

LIST 分 区 与 RANGE 分 区 类 似 ，RANGE 分 区 的 规则 是 连续 区 间 ， 而 LIST 规则 是 根 
据 指 定 的 学 段 值 是 否 与 规则 中 的 值 相 等 ， 如 果 相 等 则 创建 ， 人 否则 放弃 。 指 定 的 字段 通 第 为 
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枚 举 类 型 的 数据 。 如 以 下 代码 所 示 。 


CREATE TABLE IF NOT EXISTS ‘tpk user ( 
le@ Tme (lll) NOT NULG, 
‘user name char(20) NOT NULL, 
‘user email char(40) NOT NULL, 
‘class_id int(2) NOT NULL 
) ENGINE=MyISAM DEFAULT CHARSET=utf8 
PARTITION BY LIST (class_id) ( 
lB/AIRAL LION 90 Ek (L 273747 97 O78)y 
PARTITION 9L VALUES IN (9-10, 11, 12,16); 


); 

如 上 述 代 人 码 所 示 ， 创 建 tpk_user 数据 表 时 ， 当 classid 为 1 一 8 时 创建 pO 分 区 ; class_id 
为 9 一 16 时 创建 pl 分 区 。 也 就 是 说 ，LIST 分 区 是 必须 显 式 指定 固定 的 数值 的 ， 而 不 像 
RANGE 规则 那样 指定 一 个 数据 区 间 。 需 要 注意 的 是 ， 创 建 LIST 分 区 时 ， 数 据 表 中 不 允许 
存在 主键 索引 字段 。 

(3) HASH 分 区 

HASH 分 区 没有 特定 的 分 区 规则 ， 它 使 用 的 是 随机 分 配 的 分 区 策略 ， 在 创建 数据 表 时 ， 
只 需要 指定 PARTITIONS 数值 ，HASH 分 区 则 按照 设备 的 值 将 数据 随机 存放 到 数据 表 分 区 
中 ， 如 以 下 代码 所 示 。 


CREATE TABLE IF NOT EXISTS ‘tpk user ( 
`id`ò int (11) NOT NULL AUTO_INCREMENT, 
`user_name` char (20) NOT NULL, 
`user_email` char(40) NOT NULL, 
elass iol nme (LL) NOT NOAM, 
PRIMARY KEY (`id`) 
) ENGINE=MyISAM DEFAULT CHARSET=utf8 
PARTITION BY HASH (id) 
PARTITIONS 5; 
上 上 述 代 码 表示 创建 tpk_user 数据 表 时 ， 将 创建 5 个 HASH 分 区 。 
(4) KEY 分 区 
KEY 分 区 类 似 于 HASH 分 区 ， 但 HASH 分 区 使 用 用 户 目 定义 的 表达 式 ， 而 KEY 分 区 
使 用 的 哈 希 图 数 是 由 MySQL 服务 器 提供 。 如 以 下 代码 所 示 。 
CREATE TABLE tpk user2 ( 
“1@ Ame h(LL) NOT NUL p 
`user_name` char (20) NOT NULL, 
`user_email` char(40) NOT NULL, 
"elass iol ime (LL) NOT NULE 
el INTE NOT NULL 
) ENGINE=MyISAM DEFAULT CHARSET=utf8 
PARTITION BY LINEAR KEY (coll) 
PARTITIONS 3 


通过 前 面 的 介绍 ， 相 信 读 者 已 经 掌握 了 这 4 种 最 常见 的 分 区 技术 ， 每 种 分 区 技术 都 有 各 
自 的 特点 ， 在 实际 应 用 开发 中 可 以 灵活 选择 。 至 此 ， 前 面 提 到 的 1 亿 条 数据 的 存储 方案 相信 
读者 也 已 经 找 能 够 解决 。 最 后 还 需要 索引 步骤 ， 由 于 MySQL 调 优 不 在 本 书 讲述 范畴 ， 需 要 
读者 自行 参阅 相关 MySQL 书籍 。 

0 提示 : 要 在 数据 库 中 模拟 1 亿 条 数据 并 非 困难 的 事 ， 这 里 提供 一 种 快速 创建 数据 的 方法 。 首 先 在 数据 库 

中 创建 2 个 一 模 一 样 的 数据 表 ， 命 名 为 M t2 Ætl 表 中 手动 插入 10 条 数据 ， 然 后 在 MySQL 命令 行 
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中 运行 insert into t2 select * from tl;， 执 行 完 一 次 成 后 ， 按 键盘 上 的 “1” 键 ， 然 后 回 车 。 这 时 将 会 看 到 
t2 RA 1 倍 的 数据 量 重复 庄 加 。 连 续 重 复 同 样 操 作 ， 刀 数据 表 将 很 快 达到 存储 极限 ， 使 用 这 种 方式 可 以 
有 效 地 测试 数据 表 分 区 和 数据 表 索 引 性 能 。 


3. 分 区 管理 

在 创建 分 区 时 或 者 创建 分 区 后 可 以 对 分 区 的 文件 指定 存放 位 置 。 

(1) 改变 文件 位 置 

默认 情况 下 ， 数 据 表 分 区 文件 和 数据 表 文 件 存 放 一 起 ， 如 果 为 了 便于 管理 ， 在 创建 时 可 
以 通过 DATA DIRECTORY 选项 手动 指定 数据 表 分 区 文件 存放 位 置 ， 如 以 下 代码 所 示 。 


CREATE TABLE IF NOT EXISTS tpk user ( 
`id`ò int (11) NOT NULL AUTO_INCREMENT, 
`user_name` char (20) NOT NULL, 
`user_email` char(40) NOT NULL, 
elass Lol mme CLIL) NOT NUL 
PRIMARY KEY (`id`), 

KEY "user name (`user_name`, ` user_email`) 

) ENGINE=MyISAM DEFAULT CHARSET=utf8 

PARTITION BY RANGE (id) ( 

PARTITION p0 VALUES LESS THAN (3) 


DATA DIRECTORY = '/data0/data' 
INDEX DIRECTORY = '/datal/idx', 
PARTITION p1 VALUES LESS THAN (6) 
DATA DIRECTORY = '/data0/data' 
INDEX DIRECTORY = '/datal/idx', 
PARTITION p2 VALUES LESS THAN (9) 
DATA DIRECTORY = '/data0/data' 
INDEX DIRECTORY = '/datal/idx', 
PARTITION p3 VALUES LESS THAN (12) 
DATA DIRECTORY = '/data0/data' 
INDEX DIRECTORY = '/datal/idx', 


PARTITION p4 VALUES LESS THAN MAXVALUE 


A: 

如 上 述 代 码 所 示 ，DATA DIRECTORY 指定 分 区 数据 文件 存放 位 置 INDEX 
DIRECTORY 指定 分 区 表 索 引 数据 存放 位 置 。 

(2) 删除 分 区 

数据 表 分 区 创建 完毕 后 ， 通 名 不 需要 作 人 修改， 但 是 如 果 执 行 删除 操作 ， 可 以 使 用 drop 
删除 指定 的 数据 表 分 区 ， 命 令 如 下 。 

Meals altor EE droo PARTITION p: 

(3) 增加 分 区 

增加 分 区 和 增加 字段 类 似 ， 可 以 使 用 alter 进行 操作 ， 例 如 为 RANGE 增加 新 分 区 ， 可 
以 执行 以 下 操作 。 

mysql> alter table tpk user add PARTITION (PARTITION p4 values less than MAXVALUE) ; 

(4) 重 置 分 区 

如 果 发 现 以 前 分 配 的 分 区 不 符号 数据 存放 需求 ， 同 样 也 可 以 使 用 ater 来 重新 分 配 过 ， 
以 RANGE 分 区 为 例 ， 只 需要 重 置 PARTITION 即 可 ， 如 以 下 代码 所 示 。 


ON 
VALUES LESS THAN MAXVALUE) ; 
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数据 未 分 区 后 ， 在 应 用 层 是 不 需要 做 任何 更 改 的 ， 这 也 是 笔者 建议 该 者 使 用 数据 表 分 区 
功能 的 最 主要 原因 。 虽 然 使 用 PHP 程序 也 能 在 一 定 程度 上 实现 类 似 的 功能 ， 其 全 性 能 更 加 
好 ， 但 无 论 是 从 开发 效率 还 是 程序 的 后 期 维护 来 衡量 ， 都 不 建议 在 PHP 中 完成 。 


7.6 小结 


本 章 深 入 介绍 了 ThinkPHP 操作 数据 的 方方面面 ， 以 最 基础 的 目 定 义 模型 开始 ， 然 后 讲 
解 了 各 种 查询 言 的 使 用 。ThinkPHP 提供 了 多 种 查询 表达 式 ， 运 用 好 这 些 表达 式 是 有 效 提高 
开发 效率 的 手段 。 接 大 还 介绍 了 数据 库 高 级 开发 ， 例 如 多 数据 库 动态 切换 、 高 级 模型 、 关 联 
模型 等 重要 内 容 。 最 后 还 介绍 了 Web 3.0 中 最 典型 的 大 数据 应 用 ， 是 每 个 PHP 程序 员 都 需要 
掌握 的 内 容 。 网 站 安全 问题 无 论 何 时 都 必须 高 度 重 视 ， 下 一 章 将 深入 介绍 ThinkPHP 内 置 的 
安全 机 制 。 
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内 容 提 要 


网 站 上 线 后 安全 问题 永远 都 是 运营 企业 所 关心 的 问题 ， 网 站 的 安全 涉及 众多 方面 。 
中 最 突出 的 是 服务 器 安全 及 网 站 程序 自身 的 安全 ， 做 到 绝对 安全 是 Pa 
标 ， 但 现实 是 相对 的 。 所 以 这 就 更 需 要 开发 人 员 和 系统 管 理 人 员 认 真 对 待 每 个 细节 ， 坚 守 
职业 规范 。 

作为 开发 人 员 的 我 们 ， 最 需要 关注 的 是 程序 安全 ， 所 以 这 就 必须 要 求 在 开发 阶段 对 程序 
egener a 用 户 鉴 权 是 否 合 理 、 程 序 是 否 EEN 
问题 都 应 该 在 开发 和 测试 阶段 发 现 并 修复 。 本 章 将 结合 ThinkPHP 的 安全 机 制 ， 全 面 介 绍 
MVC 开发 的 安全 与 程序 调试 。 


学 习 目 标 


了 解 ThinkPHP 内 置 的 安全 机 制 。 

掌握 表单 令 牌 的 使 用 。 

Je F BASM. 

掌握 数据 验证 。 

了 解 ThinkPHP 安全 日 志 机 制 。 

掌握 HTTPSQS 处 理 安 全 日 志 的 实战 应 用 。 
了 解 ThinkPHP 程序 调试 机 制 。 

掌握 程序 调试 的 全 过 程 。 
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8.1 构建 稳健 及 安全 的 MVC 


接 下 来 将 讲解 ThinkPHP 内 症 的 安全 验证 机 制 ， 包 括 表单 令 牌 、 了 字段 验证 、 字 段 限 制 以 
及 日 记功 能 。 此 外 为 了 监控 数据 的 完整 性 ， 后 和 耐 还 将 介绍 HITPSQS 信息 队列 功能 ， 儿 助 读 
者 构建 真正 稳定 、 健 康 的 应 用 程序 。 下 面 首 先 来 了 解 ThinkPHP 内 年 的 安全 机 制 。 


8.2 ThinkPHP 内 置 的 安全 机 制 


ThinkPHP 内 置 了 许多 安全 处 理 机 制 ， 帮 助 开 发 者 在 应 用 层 方面 尽量 减少 安全 漏洞 。 理 解 并 
掌握 这 些 安全 机 制 是 对 每 个 ThinkPHP MVC 开发 人 员 的 基本 要 求 。ThinkPHP 内 置 的 安全 机 制 有 
表单 令 牌 、 字 段 检测 、 数 据 验证 、 数 据 完 整 性 、 数 据 验 证 但 等 功能 ， 下 面 将 分 别 进行 介绍 。 


8.2.1 表单 令 牌 


表单 令 牌 是 一 项 非常 实用 的 技术 ， 它 使 用 服务 器 的 Session 功能 ， 实 现 对 表单 数据 来 源 
的 检测 及 校 验 ， 防 止 数据 被 伪造 和 复 改 。 传 统 的 表单 提交 一 般 使 用 权限 功能 来 检测 。 例 如 在 
留言 系统 中 ， 当 用 户 提交 留言 数据 时 ， 后 全 的 程序 首先 会 局 动 基本 的 数据 校 验 程 序 ， 例 如 数 
据 是 人 否 为 衬 ， 某 个 字段 是 否 填 与 正确 等 ， 如 果 一 切 符合 条 件 ， 则 允许 提交 ， 否 则 终止 操作 。 

假设 用 户 在 站 外 ， 例 如 本 地 上 模拟 同样 的 数据 提交 给 服务 器 ， 由 于 没有 做 表单 验证 功能 ， 所 
以 这 些 数据 也 会 被 提交 ， 这 无 疑 给 网 站 留 下 了 隐 式 的 安全 隐患 ， 就 算是 使 用 传统 的 会 员 权 限 来 判 
汤 ， 如 果 没 有 数据 来 源 验 证 ， 也 是 不 可 靠 的 (用 户 可 以 通过 修改 head 等 信息 实现 cookie DON 7. 

ThinkPHP 提供 的 表单 令 脾 功能 很 好 地 解决 了 数据 来 源 校 验 ， 防 止 跨 站 提交 。 当 表单 令 
牌 开 尼 后 ， 在 存在 表单 的 视图 中 会 生成 随机 令 收 (默认 为 随机 的 MD5 字符 串 )。 当 数据 被 保 
存 到 数据 库 前 ， 开 发 人 员 可 以 使 用 Session 对 令 牌 进行 校 验 ， 如 图 8-1 所 示 。 


令 牌 验证 


创建 数据 | 


图 8-1 表单 令 牌 验证 过 程 
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表单 令 牌 默认 已 经 开局 ， 默 认 情 况 下 生成 的 表单 令 牌 使 用 随机 HASH "eer, FHF 
段 名 称 为 “HASH ， 如 以 下 代码 所 示 。 


<form action="/index.php/index/post" method="post" name="forml"> 
ul A anoue Cpe username a lie 
aea input epe pase | 


a E E a e E a A a Dames HH 

<input type="hidden" 
value="40fc857da6775ed3a56258ee250bb0ec_549de3594b50aaf7d2b634053£641d1b" name="__HASH_ "> 

EO 


如 上 述 代码 所 示 ， 当 系统 视图 引擎 遇 到 form 标签 时 ， 就 会 在 表单 后 面 追加 表单 令 牌 ， 
如 果 需 要 关闭 ， 可 以 在 当前 动作 中 配置 CCTOKEN ON false) 。 表 单 生 成 后 ， 就 可 以 在 动作 
中 验证 表单 来 源 了 ， 如 以 下 代码 所 示 。 


<?php 


class IndexAction extends Action { 
public function post (){ 
// 实例 化 User 模型 
ŞUser = M('User'); 
ŞUser=$User->create(); 
// 根据 表单 提交 的 POST 数据 创建 数据 对 象 
if (!S$SUser->autoCheckToken($ POST))I 


$this->error ("非法 的 表单 来 源 "); } 


} 


?> 


_ HASH_ 是 默认 生成 的 隐藏 字段 名 ， 开 发 人 员 可 以 通过 配置 文件 进行 更 改 。 此 外 ， 表 
单 令 牌 的 加 密 方式 等 都 是 可 以 自 定 义 的 ， 如 以 下 代码 所 示 。 


'TOKEN_ON'=>true, // 是 否 开 启 令 牌 验证 

'TOKEN_NAME'=>' hash _', // 令 牌 验证 的 表单 隐藏 字段 名 称 
'TOKEN_TYPE'=>'md5', // 令 牌 哈 希 验证 规则 默认 为 MD5 
'TOKEN_RESET'=>true, // 令 牌 验证 出 错 后 是 否 重 置 令 牌 默认 为 true 


需要 注意 的 是 ， 如 果 一 个 页 面 中 存在 多 个 表单 ， 那 么 只 能 有 一 个 表单 存在 令 牌 。 多 个 表 
单 时 ， 使 用 {__TOKEN _} 标 识 令 牌 生成 区 ， 如 以 下 代码 所 示 。 


<form name="forml" method="post" action=" URL /post"> 
D e ee Zelt 
< Ve= Ua Wr /lS 


<input name="" type="submit" value Ye 
</form> 
<form name="form2" method="post" action=" URL /post"> 


zii- E <input ype usernames" SL 
elt input type passwords" | 


El E name Y type submit value eae 
{_ TOKEN_ } 
</form> 


上 述 代码 中 ， 定 义 了 2 个 表单 ， 因 为 只 有 form2 表单 定义 了 {__ TOKEN |} 标 识 ， 所 以 
只 有 该 表单 存在 表单 令 牌 。 


8.2.2 字段 检测 
使 用 create 方法 创建 数据 系统 会 自动 进行 过 滤 和 检测 ， 但 也 可 以 单独 对 表单 中 的 单一 


190 O O 


第 8 章 安全 与 调试 


段 进行 检测 和 过 滤 。 系 统 对 单个 表单 字段 的 处 理 主 要 有 字段 过 滤 、 字 段 只 读 限制 、 字 段 类 型 
检测 等。 下面 分 别 进行 介绍 。 

1. 字段 过 滤 

学 段 过 沽 可 以 对 表单 中 的 单个 字段 进行 过 滤 ， 极 大 地 方便 了 表单 数据 处 理 。 例 如 在 数据 
插入 前 ， 开 发 人 员 可 以 对 表单 中 的 字段 进行 敏感 字 检 测 ， 在 获取 数据 时 可 以 对 单个 字段 中 的 
数据 进行 伏 换 。 字 段 过 小 只 需要 在 日 定义 模型 中 定义 $_filter 属性 即 可 (学 段 过 小 需要 继承 于 
AdvModel 模型 )。$_filter 属性 格式 如 下 。 


protected $_ filter = array ( 
' 过 小 的 字段 '=>array (' 写 入 前 处 理 函 数 ',' 读 取 前 处 理 函 数 '， 是否 传 入 整个 数据 对 象 ) ， 


) 

这 里 所 说 的 过 滤 字 段 是 指数 据 表 中 的 字段 ， 如 果 表 单 中 的 字段 与 数据 表 中 的 字段 不 一 
样 ， 则 需要 做 字段 映射 〈 孚 段 遇 射 可 参考 本 书 7.1.1 市 )。 写 入 本 处 理 函 数 和 读 取 前 处 理 函 数 
是 指 在 common.php 中 目 定 义 的 函数 。 最 后 一 个 选项 是 指 是 否 传 入 所 有 表 字 段 ， 默 认 只 传 入 
过 滤 字 段 中 定义 的 字段 ， 可 设置 的 选项 有 true 及 false。 下 面 通 过 一 个 示例 演示 $_filter 属性 
的 使 用 。 

仍然 以 tpk_article 数据 表 为 例 ， 如 末 需 要 在 数据 插入 前 ， 对 content 字段 进行 额外 处 理 。 
当 content 字段 中 存在 URL 字符 串 时 ， 华 止 用 户 输入 完整 的 URL 代码 ， 而 是 只 能 输入 一 个 
字符 串 ， 系 统 目 动 将 字符 串 转 换 成 标准 的 HTML 代码 。 同 时 在 读 取 时 系统 会 对 编辑 器 中 特 
CDI UBB 代码 进行 转换 ， 以 达到 显示 网 片 的 目的 。 

首先 让 ArticleModel 继续 于 AdvModel 高 级 模型 ， 然 后 定义 $_filter 属性 ， 如 以 下 代码 所 示 。 


<?php 
class ArticleModel extends AdvModel{ 
protected $_filter = array ( 


"Content '=>array('contentWriteFilter', 'contentReadFilter'!'), 
); 
} 


?> 


如 上 述 代 公所 示 ，contentWriteFilter 函数 是 写 入 前 的 处 理 图 数 :， contentReadFilter Pë SCH) 
于 处 理 读 取 content 字段 前 的 处 理 函 数 ， 这 2 个 处 理 函 数 均 在 Common/common.php 中 定义 ， 
代码 如 下 所 示 。 


<?php 

// 对 content 字段 写 入 前 进行 处 理 

function contentWriteFilter ($var)t 
Surft ee el ea a eae ee ee 
retruc preg replaces (url, ee 


} 
// 读 取 content 字段 前 进行 处 理 
function contentReadFilter ($var){ 
Şubbcodes=array ( 
Ee 
Ehe 
El Ge E EE 
AA Eege Eh Kebab 
EE E E E Ee 
ee I A lone 
); 
shtmls=array ( 
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EE E 
EE 
KEE? 
Ee ele EE 
'<font size="\1">\2</font>', 
<o aligns" \LvS\2</B>! 
); 
//UBB 替换 
return preg_replace (S$Subbcodes, shtmls, $var); 
} 
?> 
最 后 束 可 以 在 动作 中 进行 处 理 了 。 处 理 过 程 不 需要 开发 人 员 干 涉 ， 系 统 会 日 动 进行 四 
配 ， 完 成 学 段 过 小 的 整个 过 程 。 如 以 下 代码 所 示 。 
<?php 
class IndexAction extends Action { 


public function add(){ 
SArticle=D ("Article"); 
SdarcalYrcicleYj=$_POST| “ticle” ]z 
sdatal"content"]=$ POST["content"]; 
sdatal"adqd user"|="ceiba"; 
sdatal[l"add time"]=time (); 
$data["category"]=" 教 育 新 闻 "，; 
şdata["area"]="guandong"; 
if ($Article->add ($data) ){ 
$Sthis->success ("数据 插入 成 功 ")， 
}else{ 
$this->error ("数据 插入 失败 ")，; 
} 


} 


?> 


读者 可 以 在 表单 中 输入 url 字符 串 ， 系 统 将 目 动 转换 成 HTML 代码 。 例 如 http:/localhost 
转换 成 的 html 代码 为 <a href=http://localhost >http://localhost</a>。 这 里 只 是 作为 一 个 演示 ， 
事实 上 学 段 过 滤 在 实际 应 用 开发 中 是 非常 灵活 的 ， 可 以 应 用 在 任何 数据 处 理 场 合 。 

2. 字段 只 读 限 制 

为 了 使 数据 更 加 严谨 ， 开 有 人员 可 以 单独 对 特定 的 字段 进行 上 只 谈 限 制 。 被 限制 为 上 只 旋 的 
字段 将 不 能 接收 数据 插入 、 人 和 修改、 删除 指令 ， 只 能 用 于 显示 数据 。 只 该 字段 通 利用 于 会 员 中 
心 数 据 的 处 理 以 及 XML, SOAP 等 只 作答 询 的 场合 。 定 义 上 只 读 宇 段 需要 在 目 定 模型 中 进行 
定义 ， 并 且 需 要 继承 AdvModel 局 级 檬 型， 如 以 下 代码 所 示 。 

<?php 


class ArticleModel extends AdvModel{ 
protected $readonlyField = array ('title','add_user', 'add_time'); 


} 


?> 
定义 完 只 读 字 段 后 ， 然 后 对 tite, add user, add pm 字段 值 进 行 增 、 删 、 改 操作 时 ， 系 
统 将 停止 执行 。 


3. 字段 类 型 检测 
PHP 是 一 种 弱 类 型 脚本 语言 ， 对 变量 的 类 型 适 配 是 目 动 的 ， 也 束 是 说 PHP 对 变量 的 类 
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型 不 做 强制 区 分 ， 这 点 与 C#、Java 等 强 类 型 语言 有 着 明显 的 区 别 。 但 是 数据 表 字 段 对 类 型 
是 有 严格 限制 的 ， 所 以 为 了 提高 安全 性 ， 可 以 对 数据 字段 进行 强 类 型 检测 。 

要 开启 字段 类 型 检测 ， 需 要 在 配置 文件 中 开局 DB_FIELDTYPE_CHECK 选项 (默认 
false )。 开 局 后 ， 系 统 会 强制 检测 数据 表 字 段 与 数据 变量 之 则 的 类 型 关系 。 通 常情 况 下 ， 辽 
段 类 型 检测 用 于 数据 的 增加 和 修改 ， 以 达到 完善 数据 的 目的 。 


8.2.3 ”数据 验证 


数据 验证 是 数据 插入 数据 表 前 一 个 重要 的 步 又， 甚至 可 以 说 是 必须 要 做 的 一 个 步骤 。 不 
仅 PHP 应 用 如 此 ， 其 他 语言 类 型 的 应 用 也 如 此 。 因 为 数据 验证 关系 到 数据 的 完整 性 和 规范 
性 ， 所 以 数据 验证 需要 开发 人 员 认 真 调试 。 

传统 的 PHP 开发 ， 一 般 首 先 在 视图 用 使 用 JavaScript 或 Jquery 等 脚本 进行 初步 的 表单 验 
证 ， 然 后 在 后 台 使 用 PHP 正则 或 者 学 从 处 理 函 数 对 表单 数据 进行 验证 ， 这 无 疑 是 很 好 的 处 
理 方式 ， 但 是 由 于 编写 正则 需要 额外 的 知识 ， 并 且 如 琳 所 有 入 段 都 使 用 正则 或 函数 来 处 理 ， 
难免 会 影响 开发 效率 ， 尤 其 在 表单 字段 众多 时 更 是 值得 注意 。ThinkPHP 提供 了 人 简单 、 易 用 
的 表单 字段 验证 处 理 功 能 ， 能 够 极 大 地 提高 开发 效率 。 接 下 来 将 介绍 系统 内 置 的 验证 规则 ， 
结合 前 痛 验 证 脚本 ， 可 以 快速 地 实现 友好 、 强 大 的 数据 验证 功能 。 

1. 定义 验证 规则 

定义 数据 验证 规则 ， 需 要 在 目 定 模型 中 进行 定义 。 和 其 他 数据 处 理 方 式 一 样 ， 数 据 验证 
也 是 使 用 成 员 属 性 值 来 实现 的 ， 配 置 数据 验证 的 属性 为 $_validate， 该 属性 值 是 一 个 多 维 数 
据 ， 格 式 如 以 下 代码 所 示 。 

array (验证 字段 , 验证 规则 , 错误 提示 , [验证 条 件 ] , [附加 规则 ] , [验证 时 间 ]) 

需要 说 明 的 是 ， 系 统 提供 的 验证 方式 只 是 包装 和 简化 了 现 有 的 PHP 验证 方式 ， 例 如 
PHP 正则 、 季 人 符 串 处 理 函 数 等 ， 所 以 无 论 是 灵活 性 还 是 可 徘 性 两 者 都 是 一 样 的 。 针 对 数据 表 
单 竺 性 ， 系 统一 共 提 供 了 7 种 数据 验证 规则 ， 如 表 8-1 Hr, 


表 8-1 系统 内 置 的 表单 验证 规则 


验证 规则 


使 用 正则 进行 验证 ， 规 则 可 以 是 一 


regex array('verify',require', 验 证 人 码 必须 ! ') 个 正则 表达 式 
Ges l ERZ KEA LH or 
function array('password','checkPwd', ZIEK 2 EAO, function’ ) 使用 5 定 图 数 ， 观 则 可 以 是 一 个 日 
定义 函数 
' 11 UR ` ' H D 使 用 成 员 方 法 ， 规则 可 以 是 Model 
callback array(passSword', checkPwd', 密 码 格式 不 正确 ,0,"callback”) 类 或 当前 模型 类 中 的 成 员 方法 
We s o A DI As E R 
confirm array(Trepassword',password', 确 认 密 人 码 不 正确 ,0,”confirm” ) H UE E P O PI A A BAE iE f M 


同 ， 常 用 于 验证 两 个 字 码 是 否 相同 
SE array('verify''equal ,验证 但 必须 ! ) 验证 是 否 等 于 某 个 值 〈 绝 对 等 于 ) 
验证 字段 值 是 否 处 理 规则 中 定义 的 


> ' ' ' 4 JF q | ' I eg 
in array('value',array(1,2,3),' 值 的 范围 不 正确 ! ',2,”in”) ep, ANE i 
i 0 Ge EI 光 H A y 
unique array(name',", 账 号 名 称 己 经 存在 ! '0,'unique',1) 验证 表单 字段 值 与 数据 表 中 对 应 的 


字段 值 是 售 唯一 性 


2. 配置 验证 规则 
验证 规则 本 身 文 持 验证 错误 提示 《 文 持 多 语言 )， 所 以 在 ThinkPHP 中 使 用 表单 验证 规 
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则 非常 简单 。 为 了 方便 演示 ， 这 里 将 继续 使 用 tpk_article 和 tpk_user 数据 表 作 为 验证 数据 
表 ， 对 应 的 檬 板 表 单 代 人 码 如 下 所 示 。 


<taglib name="tp,My" /> 

<tp:editor id="textContent" uploadurl="/Public/editor up" width="600"></tp:editor> 
<form id="forml" name="forml™" method="post" action=" URL /add"> 

<div class="main"> 

过 标题 本 有 

<L> 

<textarea name="content" id="textContent" cols="45" rows="5"></textarea> 
</li> 

<input type "submit" value iaa O 

</form> 

</div> 


该 者 也 可 以 使 用 其 他 数据 表 来 演示 ， 上 只 需要 注意 字段 名 称 即 可 。 下 面 将 分 别 介 绍 系统 内 
置 的 7 种 验证 规则 。 

(1) regex 正则 验证 

正则 验证 是 系统 最 常用 的 一 种 表单 宇 段 验证 方式 ， 也 是 默认 的 验证 方式 。 例 如 判断 表单 
学 段 是 侣 为 宇 ， 系 统 使 用 的 就 是 正则 验证 。 正 则 验证 的 关键 字 为 require， 如 以 下 代码 所 示 。 

ArticleModel extends Mode) 1 


protected $_validate = array ( 
array ('title','require',，' 标 题 需 要 填写 ' ) ， 


} 


?> 


正则 验证 的 本 质 是 正则 表达 式 ， 系 统 允 许 开 发 人 员 使 用 数组 配置 代替 将 难以 记 牢 的 正则 
表达 式 ， 有 效 地 提高 开发 效率 。 

(2) 使 用 函数 验证 (function) 

正则 验证 只 适用 于 已 有 的 系统 内 置 验证 规划， 如 果 验 证 规则 不 和 存在， 验证 将 失效 。 所 以 
正则 验证 的 扩展 性 是 有 限 的 。 配 合 函 数 验 证 ， 将 能 够 实现 更 加 灵活 的 扩展 验证 。 函 数 验 证 通 
第 是 指使 用 目 定 义 函 数 实 现 验证 。 假 设 需要 验证 标题 字符 数量 ， 如 果 字 数 大 于 2 则 允许 提 
区， 人 否则 将 终止 表单 提交 。 步 又 如 下 。 

首先 需要 在 Common/common.php 中 定义 日 定义 函数 ， 该 函数 用 于 计算 中 文学 从 的 数 
量 ， 如 以 下 代码 所 示 。 

<?php 

// 计 算 中 文学 数 

function abslength ($str,) 

{ 


Sch amont = 0p 

Sen amont = 0; 
$str = preg_replace("/( | S E > 
ror(Si=0 ŞSiL<strlen (SSEE) 7 SLF) 
{ 

Sorel = orci (Ssrr rI SLi)? 

if ($ord > 128) 

$ch_amont++; 
else 
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Zen amoOnt tt: 
} 
SstrNum= ($ch amont/3) + $en_amont; 
if (ŞstrNum>=2){ 
E EB 
}else { 


EES Eer eren 


?> 
接 下 来 需要 在 ArticleModel 目 定 义 模型 中 使 用 函数 验证 规则 来 实现 title 字段 验证 ， 如 以 
下 代码 所 示 。 
<?php 
class ArticleModel extends Model{ 
protected $_validate = array ( 
array (title, eG er iim ZELT), 
array('title','absLlength'，' 标 题 不 能 少 于 2 个 字 ',0,'function')， 
) 7 
| 
?> 


如 以 上 代码 所 示 ， 函 数 验 证 规则 和 正则 验证 规则 是 可 以 同时 使 用 的 。 函 数 验 证 规则 需要 
显 式 地 指定 验证 附加 规则 《〈 即 function)， 其 中 数组 元 素 0 表示 根据 函数 的 返回 值 判断 该 验证 
规则 的 成 立 条 件 。 

(3) 使 用 方法 验证 (callback) 

使 用 callback 方法 验证 和 使 用 函数 验证 是 一 样 的 ， 不 同 之 处 在 于 callback 的 验证 附加 规 
则 是 Model 类 及 当前 类 中 的 成 员 方 法 ， 而 不 是 一 个 功能 函数 ， 如 以 下 代码 所 示 。 

<?php 

class ArticleModel extends Model)! 

protected $_validate = array ( 
array title reguire re hon eE, 
array ('title', 'abslength',' 标 题 不 能 少 于 2 个 字 ' o 'callback'), 


); 
// 计 算 中 文学 数 
function abslength ($str) 
{ 
$ch amont = 0; 
Zen amont = 0; 
$str = preg_ replace("/( | Ee 
eelere e Ee Ee SLi) 
d 
Şord = ord($str{$i}); 
Li eg G > 128) 
$ch_amont++; 
else 
$en_amont++; 
} 
$strNum=($ch amont/3) + Sen amopnt: 
if (ŞstrNum>=2){ 


return true; 
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}else { 
return false; 


} 


} 


Ss 


(4) 对 比 表 单字 段 值 (confirm ) 
在 会 员 注 册 系 统 中 ， 通 常 需 要 重复 校 验 2 次 密码 字段 ， 以 确认 用 户 所 输入 的 密码。 系统 
提供 了 confirm 验证 规则 ， 用 于 实现 表单 中 2 个 字段 的 校 验 ， 如 以 下 代码 所 示 。 


<?php 
class UserModel extends Model{ 


protected $_validate = array ( 
årray('repassword', password, Mt 0, T ceontirmii, 


1: 


(5) WIETE A Ter TE Cequal) 
equal 验证 规则 用 于 验证 指定 的 表单 字段 值 是 否 绝对 等 于 指定 的 值 ， 该 值 由 验证 规则 所 
定义 ， 如 以 下 代码 所 示 。 


<?php 
class ArticleModel extends Model! 
protected SG validate = array ( 
TI Ee: 


} 


p> 


(6) 验证 字段 值 范 围 Gn) 
in 验证 附加 规则 用 于 验证 表单 的 值 是 否 处 于 指定 的 范围 内 ， 通 利用 于 验证 表单 值 是 合 为 
数学 。 因 为 in 能 够 指定 郊 围 ， 所 以 验证 规则 可 以 是 一 个 数组 ， 如 以 下 代 人 码 所 示 。 


<?php 
class UserModel extends Model? 


protected SG validate = array ( 


array Cd arras a a E Ea n, 


j 


(7) 验证 字段 值 是 否 唯一 Cunique) 

在 会 员 注 册 系 统 中 ， 通 第 在 用 户 输入 用 户 名 或 者 邮箱 时 ， 需 要 使 用 得 询 语句 得 询 输入 的 
用 户 名 或 邮箱 是 否 已 经 存在 于 数据 表 ， 如 条 存 在 则 终止 提交 。 这 对 于 完善 数据 和 提高 数据 的 
安全 性 是 非常 有 效 的 。 使 用 附加 规则 unique 将 变 得 简单 ， 开 发 人 员 不 需要 编写 查询 代码 ， 即 
可 轻松 地 实现 字段 查询 及 校 验 ， 如 以 下 代码 所 示 。 

<?php 


/ /User 模型 
class UserModel extends Model? 


protected SG validate = arrayl( 
en Oe 
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} 


需要 注意 的 是 ， 判 断 字 段 值 是 否 唯一 ， 通 营 情 况 下 需要 配合 Ajax 实现 。 不 仅 unique 需 
要 如 此 ， 其 他 的 验证 规则 也 一 样 ， 以 提高 用 户 体验 。 

3. 使 用 验证 规则 

一 且 在 自 定 义 模 型 中 配置 验证 规则 之 后 ， 基 本 上 束 完 成 了 数据 校 验 所 需要 的 步 又。 在 动 
作 中 只 需要 使 用 D 函数 调用 上 自 定 模型 即 可 ， 系 统 在 插入 数据 前 会 自动 进行 表单 验证 。 如 以 下 
代码 所 示 。 


<?php 
// Index 控制 器 
class IndexAction extends Action { 
EE 
public function add(){ 
$ Article =D ("Article"); 


if (!$Article->create()){ 
exit ($User->getError()); 
}else{ 


$ Article->add(); 
} 


} 


?> 


需要 注意 的 是 ， 要 使 用 表单 验证 ， 需 要 使 用 create 方法 创建 数据 。 


8.2.4 ”数据 验证 码 


数据 验证 但 是 一 项 运用 非 弟 广泛 的 技术 ， 尤 其 在 登录 系统 中 数据 验证 码 无 处 不 在 。 早 期 
的 数据 验证 个 是 为 了 防止 日 动 友 帖 软件 而 采取 的 一 项 技术 ， 它 能 够 有 效 地 防止 非 人 为 的 发 帖 
行为 ， 提 高 数据 的 安全 性 和 完整 性 。 现 在 的 数据 验证 码 可 以 用 在 多 种 场合 中 ， 有 些 对 安全 性 
要 求 极 高 的 登录 系统 还 将 验证 但 技术 结合 便 件 技术 ， 实 现 更 高 级 的 应 用 。 验 证 人 码 技术 经 过 多 
年 的 发 展 ， 越 来 越 成 熟 ， 越 来 越 多 样 化 ， 比 较 常 见 的 有 数字 或 字母 验证 码 、 语 音 验证 码 、 问 
题 对 管 验证 码 、 日 期 验证 码 等 。 

接 下 来 介绍 的 验证 码 是 基于 ThinkPHP 内 置 的 验证 码 ， 它 能 够 解决 一 般 数 据 提 前 所 需要 
的 安全 验证 ， 包 插 纯 数字 验证 、 英 文字 母 验 证 ， 数 字 和 英文 混合 验证 、 中 文 验 证 等 。 验 证 人 码 
叉 可 以 按 长 短 、 大 小 写 来 区 分 ， 接 下 来 将 评 细 介绍 系统 内 置 的 验证 个 功 能 。 

1. 生成 验证 码 

验证 码 通常 是 由 GD 基础 类 库 结 合 Session 实现 的 一 组 图 片 信息 ， 所 以 要 使 用 验证 码 需 
要 确 你 PHP 已 文 持 GD JE (5.0 以 上 版 本 已 默认 文 持 )。ThinkPHP 通过 扩展 的 方式 提供 验证 
码 功 能 ， 上 所 以 在 初始 化 验证 码 时 首先 需要 引入 Image 系统 类 库 (Car 
ThinkPHP/Extend/Library/ORG/Util/Image.class.php)， 该 类 库存 放 了 所 有 关于 图 片 处 理 的 方法 
《事实 上 就 是 对 GD 基础 类 库 部 分 功能 的 封装 )。 大 体 上 验证 人 码 共 分 为 2 类 ， 即 使 用 数字 或 字 
母 的 国际 化 验证 码 ( 下 称 普通 验证 码 ) 和 使 用 中 文 汉字 的 验证 码 。 下 面 分 别 介 绍 。 

(1) 生成 普通 验证 公 
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普通 验证 码 是 最 通用 和 第 用 的 一 种 验证 三， 也 是 一 种 最 简单 易 用 的 验证 但 。 用 于 生成 普 
通 验 证 人 码 的 方法 为 buildImageVerify， 该 方法 是 Image 系统 扩展 类 库 的 成 员 静 态 方法 ， 共 有 6 
个 参数 ， 表 现形 式 为 buildImageVerify($length,$mode,$type,$width,$height,$verifyName)。 改 变 
buildImageVerify 方法 参数 可 以 改变 验证 码 的 显示 样式 。 如 表 8-2 所 示 。 


表 8-2 buildImageVerify 方法 参数 


ZS 数 说 D 默认 值 
length 验证 码 的 长 度 £ 
验证 字符 串 的 类 型 ， 共 有 5 种 类 型 : 0 AFE 1 ART 
mode 2《〈 大 写字 母 )、3( 小 写字 母 )、4《〈 带 声调 的 中 文字 母 )、5《〈 数 字 + 字 0 
母 混合 ) 
type 图 片 类 型 ， 值 可 以 为 png 和 gif png 
width 图 片 宽度 48 
height 图 片 高 度 D) 
verifyName 验证 人 码 Session 名 称 verify 


EREEREER, Dn ol A RRI ARK. Bag "TRA 
的 验证 公 ， 只 需要 将 mode 参数 设 为 5 HU nl, ul FIr o 


<?php 
class IndexAction extends Action { 
/ / ES 
public function verify (){ 
import ("ORG.Util.Image"); 
Image: :buildImageVerify(6,5,"png",100,60); 


} 


如 上 述 代码 所 示 ，import 函数 是 一 个 引入 第 三 方 或 系统 扩展 类 库 的 函数 。 引 入 Image 类 
库 之 后 ， 由 于 buildImageVerify 方法 是 一 个 静态 方法 ， 所 以 不 需要 实例 化 。 最 终 的 验证 人 码 效 
果 如 图 8-2 所 示 。 


图 8-2 ”验证 码 


(2) 生成 中 文 验证 人 码 

中 文 验证 人 码 是 一 种 防 猿 测 更 强 的 验证 码 ， 它 解决 了 普通 验证 人 码 容易 被 破解 导致 安全 性 低 
的 问题 ， 近 年 来 在 国内 的 主流 网 站 中 均 已 被 采用 ， 例 如 QQ 安全 中 心 、 狐 浪人 微 博 等 。 中 文 验 
证 码 在 国内 被 广泛 采用 的 根本 原因 是 它 与 普通 验证 码 相 比 ， 防 破解 能 力 更 强 ， 字 体 更 友好 ， 
适合 中 文 网 站 使 用 。 与 语音 或 多 媒体 验证 码 相 比 ， 它 更 直观 ， 终 端 设备 上 不 需要 特殊 要 求 。 
但 是 中 文 验 证 码 也 有 缺点 ， 最 显著 的 下 是 中 文 验 证 码 只 适用 于 面 癌 中 文 地 区 的 用 户 。 

Image 类 库 中 的 GBVerify 静态 方法 用 于 生成 中 文 验证 码 ， 使 用 该 方法 生成 中 文 验证 码 是 
基于 字库 的 ， 系 统 并 没有 目 带 字库 ， 所 以 如 果 需 要 正确 地 生成 中 文字 符 ， 首 先 需 要 字库 。 读 
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者 可 以 在 网 上 搜索 下 载 ， 或 者 在 Windows 系统 里 的 C:\Windows\Fonts 目录 中 选择 ， 不 同 的 
学 库 将 会 显示 不 一 样 的 文字 效果， 最 后 将 字库 文件 复制 到 ThinkPHP\Extend\Library\ORG\Util 
扩展 目录 中 。 完 成 上 述 步骤 后 就 可 以 使 用 GBVerify 方法 了 。 

GBVerify 方法 共有 6 个 参数 ， 形 式 为 GBVerify ($length, $type, $width, $height, $fontface, 
$verifyName)。 改 变 参数 的 值 将 改变 中 文 验证 码 的 生成 效 素 ， 如 表 8-3 所 不 。 


SS 8-3 GBVerify 方法 参数 


参 数 说 D 默 认 值 
length 验证 码 的 长 度 4 
type 图 片 类 型 ， 值 可 以 为 png M gif png 
width 图 片 宽 度 180 
height 图 片 高 度 50 
fontface 字库 文件 地 址 〈 绝 对 路 径 ) simhei.ttf 
verifyName 验证 码 Session 名 称 verify 


这 里 为 了 方便 污 示 ， 将 使 用 C:\Windows\Fonts\simhei.ttf 字库 。 有 了 中 文字 库 ， 和 后 成 中 文 
验证 码 将 变 得 非常 简单 ， 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 


public function verify (){ 
import ("ORG.Util.Image"); 
Image: :GBVerify ();}; 


} 


?> 


最 终 的 文字 是 随机 的 ， 并 且 会 被 加 密 后 存放 于 Session 中 。 效 果 如 图 8-3 所 示 。 


图 8-3 中文 验证 但 效果 


需要 说 明 的 是 ， 无 论 是 普通 验证 码 还 是 中 文 验 证 码 ，GD 库 生 成 图 片 依 赖 于 浏览 器 Head 
头 信息 ， 所 以 在 Head 之 可 不 能 有 任何 的 输出 ， 否 则 将 导致 生成 失败 。 这 也 就 意味 看 验证 人 码 动 
作 之 前 或 之 后 都 不 能 有 任何 的 数据 被 输出 ， 为 了 保险 起 见 ， 可 以 将 验证 码 定义 在 入 口 文 件 中 。 

2. 使 用 验证 码 

无 论 是 普通 验证 码 还 是 中 文 验证 码 ， 最 终 的 文件 都 是 一 个 图 片 文件 。 所 以 在 使 用 时 下 接 
为 img 标签 赋予 图 片 路 径 即 可 ， 如 以 下 代码 所 示 。 

SC 

list-style:none; 


margin: 6px; 

j 

</style> 

<script language="javascript"> 
function show (obj) { 
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obj.src=" APP _ /index/verify/random/"+Math.random (); 
} 
</ BCEE 
<form name="form1" method="post" action=" URL /post"> 


< me nme eae ee a Lis 

<li> i: <input type "password" name='password! J></li> 

<li> 验证 码 : <input type="text" name="verify" size="8"/></li><span><img sre=" _URL /verify/" 
onclick="show (this) ; "/></span> 

有 


</form> 


如 上 述 代 码 所 示 ， 目 定义 show WARAH FEMRE aE ENL 
重新 分 配 验证 码 )。 最 终 效 果 如 图 8-4 fro, 


~ Firefox" | 


Deele 


GE? 


网 站 公告 


首页 | 联系 方式 | ATEH | ŻXAE | 订阅 


Copyright by me and not one oiher person 2012 


图 8-4 ”验证 码 使 用 效果 


上 述 登 录 表 单 将 提交 到 post 动作 中 ， 在 post 动作 中 需要 对 验证 码 进行 验证 ， 以 确保 用 
户 输入 的 验证 码 有 效 和 正确 。 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 
e 
public function post () { 
if (md5 ($_POST [verify]) !=SŞ_SESSION["verify"] 


Il emety (POS Tv ry Il emptwië SESS TON ery 
Sthis--error ("HITNA ME A; 
} 
// 实例 化 User 模型 
SUser = M('User'); 
$data["user_name"]=$_POST["username"]; 
Şdata["password"]=md5 ($_POST [|"password"]); 
$rows=$User—->where ($data)->count ();}; 
if ($rows){ 
EHS -Success (CE a KI"); 
}else{ 
$this->error ("登录 失败 ")， 
} 
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8.3 ThinkPHP 安全 日 志 机 制 


日 志 是 网 络 应 用 中 必 不 可 少 的 一 种 安全 机 制 ， 用 于 记录 网 站 的 健康 状态 。 根 据 应 用 的 不 
同 ， 日 志 的 作用 和 方式 等 也 不 同 ， 成 熟 的 MVC 框架 都 提供 有 日 志 记 录 功 能 ， 有 些 还 结合 
动 报警 接口 实现 手机 和 短信、 邮件 等 安全 报警 。ThinkPHP 日 志 功 能 够 实现 应 用 程序 出 错 状 
态 、 数 据 库 操 作 状 态 、 特 殊 文件 更 改 等 日 六 的 记录 ， 这 些 都 是 系统 目 市 的 ， 开 发 人 员 只 需要 
开 司 日 六 记录 即 可 。 当 然 ，ThinkPHP 对 目 定 义 日 记 的 文 持 也 非常 完善 ， 系 统 提供 了 多 种 记 
录 方 式 ， 接 下 来 将 分 别 介 绍 。 


8.3.1 ”记录 万 式 


ThinkPHP 共有 两 大 类 型 日 志 : 一 种 是 针对 框架 本 喘 的 运行 状态 ， 另 一 种 是 针对 基于 
ThinkPHP 构建 的 应 用 日 志 ， 也 称 手动 日 志 。 这 两 类 日 志 默 认 情 况 下 都 保存 于 Runtime/Logs 
目录 。 要 开启 日 志 记 录 功 能 ， 只 和 需要 在 配置 文件 中 设置 LOG_RECORD 为 True 即 可 。 
ThinkPHP 处 理 日 志 的 基础 类 为 Log 静态 类 ， 访 类 提供 了 多 种 记录 方式 ， 如 表 8-4 所 示 。 


表 8-4 Log 类 日 志 记录 方式 


SYSTEM 使 用 PHP 引擎 内 置 的 日 志 处 理 机 制 ( 需 要 在 php.ini 中 配置 S 
error_log) 

MAIL 使 用 error_log 函数 ] 

FILE 直接 写 入 文本 文件 中 3 

SAPI 使 用 PHP 接口 


日 志 的 记录 方式 可 以 在 配置 文件 LOG_TYPE 选项 中 进行 指定 。 默 认 LOG TYPE 为 3， 
即使 用 文本 文件 记录 。 如 果 需 要 更 改 日 志 记 录 方 式 ， 根 据 表 8-4 的 值 进行 设置 即 可 。 例 如 修 
改 为 邮件 记录 方式 ， 代 码 如 下 。 


OGEEYPE YY > 
'LOG DEST' =>'kf@86055.com', 
'LOG_ EXTRA' =>'From: webmaster@example.com', 


LOG_DEST 指定 发 送 方 邮箱 ，LOG_EXTRA 指定 接收 方 邮 箱 。 需 要 注意 的 上 述 配置 只 
是 应 用 层面 的 配置 ， 如 果 需 要 确保 邮件 能 够 发 送 正常 ， 还 需要 配置 PHP 中 的 MAIL 函数 ， 
并 确保 邮件 系统 正常 。 

日 记 的 记录 时 间 格 式 默 认 使 用 全 格式 ， 读 者 可 以 修改 为 容易 理解 的 时 间 格 式 ， 例 如 常见 
的 “年 月 H 时 :分 : 秒 ” 只 需要 在 动作 中 调用 Log 静态 类 前 修改 即 可 ， 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 


public function hello(){ 
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I mat 0 ev om el 
Şuser=M ("User"); 

Şrows=$user—->where ("sid>0")->select(); 

if (empty (Şrows["name"])){ 


Log: :write ("返回 的 用 户 名 为 空 ")， 


?> 


8.3.2 系统 日 志 


系统 日 志 是 系统 内 置 的 一 种 自动 记录 的 日 志方 式 ， 它 完整 地 记录 了 系统 运行 中 的 出 错 信 
息 ， 例 如 文件 导入 失败 、 视 图 引擎 解释 错误 、CURD 操作 错误 等 。 根 据 级 别 ， 共 分 为 9 级 ， 
如 表 8-5 所 示 。 


表 8-5 系统 日 志 级 别 


级 别 说 明 
EMERG 严重 错误 ， 程 序 将 终止 运行 
ALERT 警戒 性 错误 ， 必 须 修 复 错 误 
CRIT 临界 值 错误 ， 超过 临界 值 的 错误 
ERR 一 般 性 错误 
WARN 警告 性 错误 ， 需要 发 出 警告 的 错误 
NOTICE 提示 性 通知 ， 程 序 可 以 运行 ， 但 会 提示 有 错误 存在 
INFO 信息 ， 程 序 输出 信息 
DEBUG 调试 ， 用 于 调试 信息 
SQL SQL 语句 解释 异常 


在 配置 项 LOG_LEVEL 中 配置 系统 的 日 志 级 别 ， 可 以 将 多 个 日 志 级 别 一 起 配置 ， 每 个 
级 别 使 用 “,” 隔 开 ， 如 以 下 代码 所 示 。 


'LOG LEVEL' =>"'EMERG,ALERT,CRIT,ERR,SQL', 


8.3.3 应 用 日 志 


应 用 日 志 是 在 MVC 应 用 中 开发 人 员 手 动 记录 的 日 志 。 手 动 记 录 可 以 将 需要 记录 的 信息 
保存 到 日 六 系统 中 ， 方 便 在 系统 出 错时 能 够 第 一 时 间 排 租 故 障 所 在 。 手 动 记 录 日 志 虽 然 厅 
烦 ， 但 却 是 保证 网 站 稳定 、 安 全 运行 的 关键 ， 所 以 在 实际 应 用 开发 中 ， 应 该 要 养 成 民 好 的 日 
忘记 录 习 惯 ， 无 论 对 于 系统 维护 人 员 还 是 后 期 的 网 站 维护 人 员 都 是 友好 并 重要 的 。 

Log 类 提供 了 两 种 手动 记录 日 忘 的 方式 : 一 种 为 立即 写 入 ; 画 一 种 分 步 与 入 。 无 论 哪 一 
种 方式 ， 对 于 开发 人 员 而 言 都 是 简单 匈 用 的 。 下 面 分 别 介绍 。 

1. 立即 与 入 

立即 写 入 是 一 种 比较 传统 的 日 志 记 录 方 式 ， 如 果 使 用 FILE 记录 方式 ， 立 即 写 入 就 等 于 
D'VISITE, HI D AJ) write 静态 方法 实现 ， 该 方法 共有 5 个 参数 ， 表 现形 式 为 
Log::write($message,$level, $type=",$destination=",$extra=")， 参 数 作 用 如 表 8-6 所 示 。 


202 DO E 


第 8 章 安全 与 调试 
SS 8-6 write 静态 方法 参数 
5 数 说 明 Wi A J8 
message 手动 记录 的 日 志 内 容 SE 
level 日 志 级 别 ERR 
type 记录 方式 ， 受 LOG_TYPE 配置 项 影响 全 
destination 日 志 目 标 ， 受 LOG_DEST 配置 项 影响 Ss 
extra 额外 日 志 内 容 ， 受 LOG_EXTRA 配置 项 影响 空 
除了 参数 message 是 必 选 项 之 外 ， 其 余 参 数 都 为 可 选项 。 所 以 最 简单 的 手动 记录 日 志 代 


但 如 下 所 示 。 
ŞgetLastSql=Ş$user->getLastsql(); 
Te i ee ee 
2. 分 步 写 入 


分 步 写 入 可 以 在 一 定 程 度 上 解决 立即 写 入 导致 IO 开销 过 大 的 问题 。 分 步 写 入 最 大 的 特 


氮 是 首先 将 日 六 信 息 暂 存 内 存 ， 在 这 一 过 程 中 程序 的 运行 不 受 影响 ， 由 于 可 以 避免 CURD 
操作 繁忙 阶段 ， 所 以 分 步 写 入 效率 更 加 融 。 分 步 写 入 共 分 为 两 个 步 召 。 首 先 使 用 record 方法 


将 日 志 写 到 内 存 中 ， 丰 到 开 怪 人 员 和 手动 调 用 save 方法 时 日 志 信 息 


(1) record 方法 


才 最 终 被 水 久保 存 。 


record 方法 用 于 定义 日 志 ， 例 如 日 志 的 信息 、 级 别 等 。record 将 收集 到 的 信息 保存 到 数 


组 中 ， 在 这 一 过 程 中 数组 中 的 信息 是 临时 性 的 ， 程 序 可 以 继续 运行 ， 


也 可 以 继续 往 数 组 中 添 


加 日 志 信 息 。record 方法 共 文 持 3 个 参数 ， 表 现形 式 为 record($message,$level=self:: 


ERR,$record=false)。 改 变 record 方法 参数 值 ， 可 以 改变 日 六 的 记录 方式 ， 如 表 8-7 所 示 。 


A 认 值 


KEN 


SE 


表 8-7 record 方法 参数 
参数 说 明 
message 日 志 内 容 
level 记录 级 别 ， 受 LOG_LEVEL 配置 项 影响 


record 


ERR 


false 


是 否 强 制 记录 ， 受 LOG_LEVEL 配置 项 影响 


(2) save 方法 
save 方法 不 允许 单独 使 用 ， 需 要 配合 record 方法 一 起 使 用 。 


一 旦 调用 save 方法 ,日 志 


将 会 立即 执行 保存 功能 ， 该 方法 支持 3 个 参数 ， 表 现形 式 为 Log::save($type="， 


$destination=",$extra=")， 参 数值 如 表 8-8 Dirr, 


表 8-8 save 方法 参数 


参数 说 明 


type 日 志 类 型 ， 受 LOG_LEVEL 配置 项 影响 
level 志 记 录 日 标 ， 受 LOG LEVEL 配置 项 影响 
record 


2 认 值 


Sr 


le: 


ERR 


TURIS, % LOG LEVEL 配置 项 影响 


false 
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因为 分 步 写 入 日 忘 古 站 先 暂时 存放 的 ， 这 束 意 味 看 一 次 可 以 写 入 多 条 日 忘记 录 ， 如 以 下 
代码 所 示 。 


public function hello(){ 
Şuser=M ("User"); 
Ş$rows=$user->where ("sid>0")->select(); 
ŞgetLastSsql=Ş$user->getLastsql(); 
和 
Log: :record(' 测 试 调试 错误 信息 '，Log: :DEBUG)， 


Log: :save ()，; 


} 

ThinkPHP 提供 的 日 记功 能 比较 传统 ， 但 开发 人 员 可 以 重 写 Log 类 实现 更 加 完善 的 日 记 
系统 。 例 如 将 日 志保 存 到 数据 库 、Memcache、 消 息 队 列 、 分 布 式 文件 系统 、 数 据 云 中 等 ， 
这 些 都 可 以 通过 重 写 Log 类 实现 。 当 然 更 简单 的 方法 是 通过 系统 提供 的 扩展 机 制 来 实现 ， 关 
于 扩展 功能 的 介绍 ， 本 书 第 11 章 将 会 全 面 介绍 。 


8.4 使 用 消息 队列 机 制 


消息 队列 是 一 个 使 用 寞 步 处 理 的 数据 处 理 引 擎 ， 使 用 消 县 队列 不 仅 能 够 提高 系统 的 负 
人 和 傈 ， 还 能 够 改善 因 网 络 阻塞 导致 的 数据 缺失 ， 在 大 型 的 互联 网 应 用 中 消息 队列 应 用 得 非常 广 
沁 。 通 常用 于 邮件 发 送 、 手 机 短信 和 发送、 数据 表单 提交 、 图 片 生 成 、 视 频 转 换 、 日 志 储 存 
等 。 由 于 使 用 的 是 寞 步 处 理 ， 所 以 消 居 队 列 特别 适合 于 瞬间 数据 量 超 大 的 网 站 。 消 恩 队 列 很 
早 驶 被 国内 外 的 大 型 网 站 所 采用 ， 提 供 消 息 队 列 服务 的 软件 ， 经 过 多 年 的 发 展 已 经 变 得 非常 
稳定 和 成 熟 ， 无 论 是 在 Windows Server 还 是 Linux 系统 ， 都 有 相应 的 解决 方案 (Windows 
Server 默认 已 经 内 置 )。 在 Linx 平台 上 使 用 的 最 广泛 的 有 ZeroMQ、Posix、SquirrelMQ、 
Redis, QDBM, Tokyo Tyrant 等 ， 并 且 这 些 软件 都 是 开源 和 免费 的 。 

接 下 来 将 介绍 一 到 性 能 优 民 并 由 国内 开发 者 开发 并 维护 的 消 居 队列 开源 软件 ， 该 软件 命 
名 为 HTTPSQS。 顾 名 思 义 ， 这 球 软 件 是 基于 HTTP 协议 来 实现 消息 队列 的 ， 所 以 无 论 在 使 
用 和 调试 方面 都 是 透明 的 ， 非 党 便于 开 及 人员 进 行 系统 整合 ， 同 时 作者 还 提供 了 PHP、 
Java, Perl, Shell, Python, Ruby 等 开发 接口 ， 使 得 开发 消息 队列 应 用 变 得 简单 和 快速 。 接 
下 来 将 全 面 介 绍 消 息 队 列 相 关 知 识 并 结合 MVC 框架 ， 演 示 消 息 队 列 的 实战 应 用 。 


8.4.1 HTTPSQS 基础 


HTTPSQS 是 一 款 用 于 解决 消息 队列 的 开源 软件 ， 能 够 高 效 地 运行 在 BSD、Linux 等 服 
Fr Eo HTTPSQS 拥有 Tokyo Tyrant 数据 持久 化 的 特性 ， 也 拥有 Memcache 大 内 存储 存 的 
功能 ， 极 大 地 提高 了 数据 完整 性 和 队列 知 吐 量 。 与 Memcache 一 样 ，HTTPSQS 也 是 使 用 异 
步 事件 触发 来 唤醒 进程 的 ， 能 够 确保 数据 及 时 入 队 和 出 队 ， 如 图 8-4 rar, 

图 8-4 中 ， 数 据 首 先 由 接口 提交 给 HTTPSQS 队列 服务 器 ，HTTPSQS 内 置 有 过 小 器 ， 
它 能 够 对 数据 进行 验证 或 加 密 ; 然后 将 数据 按 移 来 先 到 的 规则 从 低 到 高 进行 列队 ， 并 且 为 每 
个 队列 添加 唯一 的 pos 值 ， 一 个 队列 占 一 块 储存 区 (相同 于 数据 表 )， 一 个 队列 里 最 多 可 以 
存放 10 亿 条 数据 。 
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图 8-4 HTTPSQS 队列 数据 过 程 


数据 出 队 时 ， 由 先入 先 出 的 规则 进行 出 队 ， 出 队 时 知 要 经 过 验证 右 ， 验 证 右 会 对 队列 进 
行 校 验 ， 例 如 密码 是 否 正 人 确 和 等。 最 后 由 MVC 程序 处 理 出 队 的 数据 ， 例 如 将 数据 永久 存放 到 
数据 库 、 发 送 到 邮件 系统 、 手 机 短信 系统 等 ， 完 成 整个 消 四 队 列 的 过 程 。 

需要 说 明 的 是 ， 在 数据 入 队 后 HTTPSQS 并 不 能 监控 数据 何 时 到 达 ， 所 以 并 不 能 日 动 将 
数据 推出 队列 ， 但 是 使 用 Linux 的 计划 任务 或 者 监控 脚本 ， 即 可 实现 目 动 出 队 。 与 国外 成 熟 
的 消息 队列 软件 相 比 ，HTITPSQS 为 了 提高 性 能 人 徐 化 了 过 滤器 、 验 证 右 并 且 只 提供 单一 的 出 
入 队 方式 ， 使 得 整个 软件 体积 不 到 900KB ， 对 于 追求 性 能 的 网 站 再 合适 不 过 了 。 关 于 
HTTPSQS 的 更 多 介绍 ， 读 者 可 以 和 丛 看 项 目 托管 地 址 http://code.google.com/p/httpsqs/。 接 下 来 
将 全 面 介绍 HTTPSQS 的 实战 应 用 ， 本 市 内 容 需 要 读者 掌握 基本 的 Linux 网 络 知识 。 


8.4.2 SS HTTPSQS 


HTTPSQS 只 有 Linux 源 代码 包 ， 没 有 Windows 安装 包 ， 也 没有 RPM 包 。HTTPSQS 可 
以 安 北 在 主流 的 Linux 发 行 版 本 或 者 BSD 操作 系统 上 。 接 下 来 在 Centos 6.0 操作 系统 上 安 
装 HTTPSQS 1.7， 读 者 可 以 在 虚拟 机 上 搭建 系统 环境 (关于 虚拟 机 的 使 用 ， 读 者 可 以 网 上 搜 
索 ， 建 议 使 用 免费 的 VirtualBox)， 但 需要 确保 系统 能 够 正常 连接 五 联网 。 

1. 安 交 前 准备 

HTTPSQS 只 有 源码 包 ， 所 以 在 安装 前 需要 确保 系统 已 经 安装 好 了 wget 下 载 工 具 及 
gcc, make 编译 环境 。 此 外 ，HTTPSQS 核心 引擎 是 基于 TokyoCabinet 核心 包 的 ， 
TokyoCabinet 是 一 个 缓存 器 引擎 ， 类 似 于 Memcached， 这 两 个 引擎 都 是 以 异步 事件 来 监听 进 
程 的 ， 所 以 在 安装 HTTPSQS 前 同样 也 需要 安装 libevent 异步 事件 依赖 库 及 TokyoCabinet 核 
心 引 擎 。 下 面 将 分 别 介绍 整个 安装 过 程 。 

(1) 安装 libevent 

为 了 你 证 安装 顺利 ， 首 先 使 用 yum ZRRFFK libevent 依赖 库 ， 命 令 如 下 。 


[root@~]# yum install zlib zlib-devel glibc glibc-devel glib2 glikb2-devel bzip2 


bzip2-devel 


然后 使 用 wget 或 curl 下 载 libevent。 
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[root@~] 


[root@~] 


[root@~] 


#mkdir /home/soft/ 
#cd /home/soft/ 


#wget http://httpsqs.googlecode.com/files/libevent-2.0.12-stable.tar.gz 


tar zxvf libevent-2.0.12-stable.tar.gz 


cd libevent-2.0.12-stable/ 


接着 解 压 libevent-2.0.12-stable.tar.gz 源码 包 ， 进 行 编译 安装 。 
[root@~]#./configure --prefix=/usr/local/libevent-2.0.12-stable/ 


[root@a~]# make 


[root@~]# make install 


[root@~]# cd ../ 


安装 完成 后 ， 可 以 在 /usr/local/libevent-2.0.12-stable/ 目 录 中 查看 到 编译 后 的 类 库 文件 。 


(2) Ze 


TokyoCabinet 


同样 ，TokyoCabinet 只 有 源 代 人 码 包 ， 所 以 只 能 手动 编译 。TokyoCabinet 的 安装 非常 简 


只 需要 直接 执行 编 详 步 又 即 可 。 首 先 使 用 wget 下 载 TokyoCabinet， 命 令 如 下 。 


[FootQ~] 


[FootQ~] 


#cd /home/soft/ 


#wget http://httpsqs.googlecode.com/files/tokyocabinet-1.4.47.tar.gz 


接着 解 压 tokyocabinet-1.4.47.tar.gz FIH E, MÁTE 2, 


[root@~]#tar zxvf tokyocabinet-1.4.47.tar.gz 

[root@~]#cd tokyocabinet-1.4.47/ 

[root@~]#./configure --enable-off64 --prefix=/usr/local/tokyocabinet-1.4.47/ 
[root@~]#make 

[root@~]#make install 

[root@~]#cd ../ 

安装 完成 后 ， 可 以 在 /usr/local/tokyocabinet-1.4.47/ 目 录 中 找到 相关 文件 。 完 成 上 述 步 又 


束 可 以 进行 安 站 HITPSQS T. 


h Jt 


2. UR 


HTTPSQS 


HTTPSQS 的 安装 与 tokyocabinet ~$, H mi 08 H m GTA) DI ol. 


[root@~] 
[root@~] 
[root@~] 
[root@~] 
[root@~] 


[root@~] 


[root@~] 


#cd /home/soft 

#wget http://httpsqs.googlecode.com/files/httpsqs-1.7.tar.gz 
#tar zxvf httpsqs-1.7.tar.gz 

#cd httpsqs-1.7/ 

#make 

#make install 


#cd ../ 


通过 上 述 步 又，HTTPSQS 就 安装 完成 了 。 最 后 只 需要 局 动 HTTPSQS 服务 进程 即 可 。 


[root@~] 


[root@~] 


[root@~] 


#mkdir /data0/queue 
#ulimit -SHn 65535 


#httpsqs -d -p 1218 -x /data0/queue 


httpsqs 进程 启动 后 ， 队 列 持久 化 数据 将 被 存放 于 /data0/queue 目录 中 。 为 了 便于 测试 ， 
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还 需要 将 1218 端口 设置 为 允许 穿 过 防火 场 ， 或 者 暂时 关闭 防火 墙 。 

[root@~]#service iptables stop 

此 时 可 以 直接 通过 浏览 器 访问 HITPSQS， 地 址 为 HITPSQS 所 在 的 主机 地 址 加 上 1218 
端口 号 ， 例 如 http:/92.168.2.1$:1218 。 因为 HTTPSQS 队列 为 室 ， 所 以 返回 结果 为 
HTTPSQS_ERROR。 检查 HTTPSQS 是 否 已 经 成 功 运行 ， 可 以 查看 是 和 否 存在 主 进程 ， 如 以 下 
代码 所 示 。 


[root@bogon soft]# pstree 


LaLe npr anacron 
| (ee 
—avahi-daemon 
—console-kit-dae 
上 一 czond 
—dbus-daemon 
—httpsqs httpsqs 
—login bash 
—master—r— pickup 
| ee 
—5* [mingetty] 
—rsyslogd 
—sshd—r bash 
| = 


sshd 


KS 


avahi-daemon 
63*[{console-kit-da}] 


{dbus-daemon} 
{httpsqs} 


2*[{rsyslogd}] 
EE 


sftp-server 


2* [udevd] 
只 需要 使 用 pkilll 关闭 进程 名 称 即 可 关闭 HTTPSQS。 在 启动 HTTPSQS 时 ， 最 常用 的 局 
动 参数 为 -4、-p、-x， 事 实 上 HTTPSQS 还 有 许多 可 选 的 参数 ， 如 表 8-9 所 示 。 


udevd: 


表 8-9 HTTPSQS 参数 


Z 数 说 D S 认 值 

-p 可 选 ， 监 听 的 tpe žm O +y 1218 

Si 必须 ， 持 久 化 队列 数据 存放 目录 

-$ HX, F NFE A A EAA E E ër J 

-C 可 选 ， 内 存 中 绥 存 的 最 大 非 叶 子 市 点 数 1024 

-m 梧 选 ， 允 许 临时 性 队列 占用 多 大 内 存 〈M 为 单位 ) 100 

-i 可 选 ，HTTPSQS 进程 号 存放 文件 /tmp/httpsqs.pid 
-a 可 选 ， 访 问 HTTPSQS 服务 器 密码 T 

-d 可 选 ， 强 制 以 守护 进程 运行 全 

-h 可 选 ， 显 示 帮 助 


8.4.3 Jm HTTPSQS 


为 了 方便 讲解 ， 帮 助 读 者 加 深 对 HTTPSQS 的 直观 认识 ， 下 面 将 对 前 面 安 装 完成 的 
HTTPSQS 进行 简单 的 测试 。 测 试 HTTPSQS 只 需要 使 用 浏览 器 即 可 ， 如 果 使 用 虚拟 机 ， 首 
先 需 要 确保 本 地 机 噩 能 够 连接 到 虚拟 机 上 。 消 息 队 列 主要 分 为 两 大 步骤 ， 即 入 队 及 出 队 ， 下 


面 分 别 介 绍 。 
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1. 入 队 (put) 

入 队 是 消息 列队 中 最 基本 的 操作 。HTTPSQS 只 支持 使 用 HTTP 进行 数据 提交 ， 所 以 开 
发 人 员 只 需要 使 用 浏览 器 就 可 以 向 HTTPSQS 提交 数据 。HTTPSQS 是 基于 异步 事件 的 ， 所 
有 数据 的 入 队 都 是 单一 同 回 的 ， 并 且 数 据 入 队 后 均 需 要 排序 。 入 队 网 址 格式 如 下 。 

http://host:1218/?2name= 队 列 名 opt=put&zdata= 数 据 &auth= 加 和 密 口 令 

队列 名 是 一 个 数据 集 的 名 称 ， 通 常情 况 下 可 以 使 用 一 个 数据 表 名 称 或 者 日 期 时 间 等 ; 
opt 是 操作 的 标识 ; data 表示 入 队 的 数据 ， 一 次 只 能 入 队 一 条 数据 ， 数 据 入 队 后 系统 会 自动 
分 配 pos 值 (相当 于 自动 增加 的 di, auth 表示 当前 队列 的 加 密 口 令 。 

HTTPSQS 能 够 接受 GET 或 者 POST 提交 方式 。 需 要 注意 的 是 ，data 参数 如 果 为 中 文 ， 
需要 进行 URL 转 码 。 加 密 口 令 不 是 必需 的 ， 如 条 HTTPSQS 服务 器 处 于 内 网 ， 建 议 放 弃 队 
列 加 密 ， 这 样 将 获得 更 好 的 运行 效率 。 下 面 将 通过 示例 演示 入 队 的 实际 操作 ， 如 以 下 代码 所 
~ http://192.168.2.15: 1218/?name=testhttpsqs&opt=put&data=hellohttpsqsg&auth=123456 


如 果 入 队 成 功 ， 浏 览 器 将 返回 HITPSQS_PUT_OK。 同 时 使 用 Firefox Firebug 插件 查看 
head 头 信息 ， 可 以 查看 到 HTTPSQS 返回 的 pos 值 ， 该 值 即 为 当前 数据 存放 在 队列 中 的 具体 
位 置 。 读 者 可 以 改变 data 参数 数据 ， 并 多 次 刷新 ， 观 察 返回 的 pos 值 ， 如 图 8-5 所 未。 


中 | 清除 保持 | 所 有 | HTML Ces JS XHR 图 片 Flash 媒体 


Content-Type text/plain 
Date Thu, 18 Oct 2012 20-01:43 GMT 


Pos 6 
请求 头 信息 Idee 
Accept Test application xhim+xml application’ zsm;g=0.9,*"g=0.8 

Accept-Encoding gzip, deflate 
Accept-Language zh-cn zhg=08en-usg=05eng=0.3 

Comecon keep-alrmve 

Host 192 168 .215-1218 
User-Agent Mozllarp 0 (Windows NT 61; rv160 Geckor/20L00101 Frefoxw16.0 


图 8-5 HTTPSQS 入 队 返 回信 息 


O 提示: 入 队 时 HITPSQS 返回 值 共 有 3 种 状态 : HTTPSQS_PUT_OK 表示 入 队 成 功 ; HTTPSQS_PUT_ERROR 
表示 入 队 失 败 ， 需 要 检查 提交 的 格式 是 否 正确 ; HTTPSQS_PUT_ERROR 表示 队列 已 满 ， 单 个 队列 的 默认 值 
为 100 万 条 ， 理 论 值 最 大 可 存放 10 亿 条 ， 更 改 入 队 参 数 num 可 以 改变 队列 数据 条 数 。 


2. 出 队 (get) 

可 以 使 用 HTTP 进行 入 队 ， 当 然 也 可 以 使 用 HTTP 获取 队列 数据 (出 队 )。HTTPSQS 文 
持 出 队 时 对 数据 进行 序列 化 ， 默 认 情 况 下 为 TXT 文本 。 出 队 的 URL 格式 如 下 所 示 。 

http:/host:1218/?charset=utf-8&name= 队 列 名 称 &opt=get&auth= 队 列 口 令 

入 队 和 出 队 是 成 正比 的 ， 也 束 是 说 先入 队 的 数据 就 会 先 出 队 。 出 队 时 opt 参数 值 共有 两 
D: get 表示 返回 txt 类 型 数据 ，status_json 表示 使 用 json 序列 化 数据 。 如 以 下 代码 所 示 。 
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http://192.168.2.15: 1218/?charset=utf-8&name=testhttpsqs&opt=get&auth=123456 
WR BAI, Mikel json Pal Zepi aAa’ TUK S i ie 

HTTPSQS_GET_END 异 剃 信息， 表示 数据 不 存在 。 同 时 使 用 Firefox Firebug HAA head 
头 信 息 ， 可 以 但 看 到 当前 数据 的 pos 值 ， 该 值 总 是 由 小 到 大 进行 变化 的 。 如 图 8-6 PTR. 


n| 清除 wen |[ 所 有 | HTML Ces JS XHR 图 片 Flash SI 


E GET harset vi? Bämaen pt tt ZO CR 


SI 加 一 
ER tet/him appicalionxhimi o emm appetaboeveend og - A "= 
Accoept-Enomling grip, deflate 
Zccept-Lasamaeger h-on cho Ben-us übe 
| = 


图 8-6 HTTPSQS 出 队 返 回信 息 


数据 出 队 是 排序 进行 的 ， 一 旦 出 队 ， 数 据 即 会 从 HITPSQS 数据 库 中 删除 。 如 果 不 需 市 
除数 据 ， 而 只 是 想 碍 看 数据 ， 可 以 指定 pos 值得 阅 对 应 的 数据 ， 如 以 下 代码 所 示 。 


http://192.168.2.15:1218/?charset=utf-8&name=testhttpsqs&opt=view&pos=12&auth=123456 


8.4.4 在 MVC 中 使 用 HTTPSQS 


熟悉 Web 开发 的 读者 经 过 前 面 介绍 的 HTTPSQS 测试 ， 相 信和 已 经 了 解 了 怎样 在 PHP 中 
提交 数据 入 队 。PHP 本 喘 内 置 的 file_get_content 或 者 CURL RAAR SKEL HTTPSQS 数 
据 入 队 。 在 应 用 程序 层面 ， 主 要 涉及 入 队 ， 数 据 出 队 通 常情 况 下 不 需要 程序 获取 ， 因 为 消息 
队列 的 本 质 就 是 目 动 提 交 数 据 ， 如 果 使 用 程序 来 获取 消 恩 队列 中 的 数据 ， 那 就 没有 使 用 消 轧 
队列 的 必要 。 接 下 来 将 介绍 在 MVC 中 入 队 数 据 ， 然 后 再 介绍 利用 脚本 功能 实现 消 因 队列 日 
动 出 队 。 

1. 入 队 

HTTPSQS 入 队 使 用 的 是 HTTP 提交 ，PHP 内 置 了 许多 HTTP 功能 函数 ， 方 便 开 发 人 员 
选择 。 同 时 ，HTTPSQS 还 提供 了 多 种 语言 的 类 库 ， 用 于 实现 HTTPSQS 入 队 及 出 队 的 所 有 
操作 ， 由 于 目前 还 没有 介绍 ThinkPHP 扩展 ， 所 以 这 里 不 使 用 作者 提供 的 类 库 ， 而 是 使 用 
PHP 内 置 的 CURL 函数 。 为 了 便于 开 及 ， 只 需要 对 CURL 进行 何 单 地 封 北 妈 可， 如 以 下 代 
公所 示 。 


<?php 
//curl 提交 数据 


function curl post ($url, Spost data=array ()){ 


Sen = CHIE) ee 

Cul GE EE ÉIER le 

curl_setopt ($ch, CURLOPT_RETURNTRANSFER, 1); 
curl_setopt ($ch, CURLOPT_POST, 1); 

curl_setopt ($ch, CURLOPT_POSTFIELDS, Şpost_data); 
Sourotr = curl exee (Sen)? 

curl_close($ch); 
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return $output; 
} 


?> 


只 需要 将 上 述 函 数 写 到 Common/common.php 目 定 义 函 数 库 中 ， 即 可 在 控制 器 动作 中 调 
用 。 这 里 将 使 用 HITPSQS 处 理 日 志 信 息 ， 如 以 下 代码 所 示 。 


<?php 
class IndexAction extends Action { 
EE 
public function hello(){ 
Şuser=M ("User"); 
Şrows=$user—->where ("sid>0")->select(); 
ŞgetLastsql=Ş$user->getLastsql(); 
有 
if (empty ($rows["id"])){ 


/ /保存 日 志 记 录 
curl_post ("http://192.168.2.15:1218/?name=log&opt=put", $data); 


}else{ 
/ RAE 
1 
} 
| 


通过 前 面 的 步 又， 一 个 简单 的 消息 队列 日 志 系统 就 完成 了 。 使 用 这 种 方式 来 保存 日 志 ， 
系统 不 仅 运行 得 更 加 高 效 ， 而 且 数 据 更 加 完整 。 读 者 可 以 在 浏览 器 中 查看 队列 是 否 成 功 ， 如 
以 下 代码 所 示 。 


http://192.168.2.15:1218/?charset=utf-8&name=log&opt=get 


接 下 来 就 可 以 在 出 队 中 将 日 志 记 录 保 存 到 tpk_log 数据 表 ， 实 现 永 久保 存 。 

2. 出 队 

要 使 用 入 队 后 的 数据 就 需要 将 数据 出 队 。 例 如 将 日 志 记 录 保 存 ， 方 便 网 站 后 人 台 人 员 查 
阅 ， 就 需要 创建 存放 日 志 的 数据 表 ， 并 且 在 控制 器 动作 中 创建 相应 的 提交 动作 。 这 里 将 把 保 
存 日 志 的 数据 表 命 名 为 tpk log， 结 构 如 网 8-7 所 示 。 


SE 类 型 属性 空 SO 额外 
C] id int{11) A ZC auto_increment 
| | log title varchar{40) utf8 estonian_ci Zi 态 
| | leg content text utf8 estonlan ci E 大 
| | log time int{11) A oO 
| | status tinyint(2) ch e 


图 8-7 mk log 数据 表 结 构 
然后 在 Index 控制 右 中 创建 提交 动作 ， 并 命名 为 Save og, Tim Fran, 


<?php 
class IndexAction extends Action { 
/ /保存 日 志 
public function SaveLog(){ 
/7 安全 验证 
i (i Es OE on en Te 


EE ("no") y 


} 
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Şlog=M ("Log"); 


sdatal llog titie] Tranan"; 
Scarcal Y loc content |=wrlcdecoce ($_POST Ee EES 
şdata["log_time"]=time (); 


if ($log->add ($data) ){ 
echo "yes"; 
}else{ 


Ee rees 


至 此 ， 保 存 日 志 所 需要 的 步骤 就 完成 了 ， 接 下 来 就 需要 将 HTTPSQS 中 的 队列 数据 发 送 
给 SaveLog 动作 。 前 面 已 经 提 到 过 ，HTTPSQS 本 身 没 有 自动 提交 数据 《〈 出 队 ) 的 功能 ， 需 
要 开发 人 员 自 行 编写 监控 脚本 。 在 Linux 系统 中 ， 可 以 使 用 官方 提供 的 C 语言 接口 ， 完 成 数 


据 的 检索 和 提交 ，C 语言 接口 下 载 地 址 为 http://code.google.com/p/httpsqs/source/browse/trunk/ 


client/c。 


对 于 只 熟悉 PH 的 开发 人 员 而 言 ， 同 样 也 提供 了 PHP 监控 脚本 ， 前 提 是 监控 服务 器 上 已 


AuR S 


PHP BOES AR. XE PHP 安装 路 径 为 /usr/local/php， 那 么 PHP DUR ARM rt AN 


/usr/local/php/bin/php。 下 和 面 将 使 用 PHP os Ze Tei HTTPSQS 监控 脚本 。PHP 类 库 
下 载 地 址 为 http://code.google.com/p/httpsgqs/source/browse/trunk/client/php/httpsqs_client.php。 

将 下 载 后 的 httpsqs_client.php 文件 保存 到 监控 服务 器 的 /opt/php_shell 目录 中 ， 然 后 使 用 
vi 等 编辑 占 创 建 一 个 监控 脚本 ， 并 命名 为 httpsqs.php， 代 人 码 如 下 所 示 。 


<?php 


include_once dirname( FILE )."httpsqs_client.php"; 
STEE "L192. 1682 115%} 

EE 

shttpsqs = new httpsqs ($host, $port, Ş$auth, Şcharset); 
while (true) { 


} 


sresult = Şhttpsqs->gets ($name); 
spos = Sreoulel "Bos / /I 人 a 
Sdata 内存 


if ($data != "HTTPSQOS GET _ END" && Sdata != "HTTPSQOS ERROR"™) { 

/ /执行 数据 提交 

$content ["data"]=$data; 

Scontene ET kay ]=Yabe 1234; 

curl_post ("http://192.168.2.1/index.php/Index/SaveLog", $content); 
} else { 


sleep (1); // 暂 俘 1 秒 钟 后 ， 再 次 循环 


function curl_post (url, Spost data=array ()){ 


Sch = GUEL lmlE()? 
SE lee Se EE EE EE pl 
curl_setopt ($ch, CURLOPT_RETURNTRANSFER, 1); 


curl secoort (Seha, CURLORT POST, 1) 
curl_setopt ($ch, CURLOPT POSTFIELDS, Spost_data); 


Soutput = curl exec ($ch); 


curl close (sch)，; 


return $output; 
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最 后 只 需要 将 httpsqs.php 脚本 推送 到 后 台 运 行 即 可 。 如 以 下 代码 所 示 。 

nohup /usr/local/php/bin/php /opt/php_shell/httpsqs.php 2>&1 > /dev/null & 

至 此， 一 个 简单 的 消息 队 列 日 志 系 统 驶 完成 了 。 通 过 消息 队列 提交 数据 ， 虽 然 驮 烦 ， 但 
市 来 的 性 能 提升 是 明显 的 。 这 里 只 是 做 一 个 简单 的 日 六 系统 ， 事 实 上 HTTPSQS 文 持 大 数 
据 ， 能 够 使 用 在 任何 需要 异步 处 理 的 场合 ， 在 实际 应 用 开发 中 可 根据 需要 进行 选择 。 

需要 说 明 的 是 ， 虽 然 消 恩 队 列 能 够 提升 性 能 ， 但 是 由 于 使 用 的 是 异步 处 理 ， 所 有 数据 都 
是 按 先入 先 出 的 排序 方式 进行 出 队 的 。 所 以 消 恩 队列 中 的 数据 始终 是 有 延迟 的 ， 如 来 需要 即 
时 性 的 结束 ， 残 不 太 适 合 使 用 消息 队列 机 制 了 。 


8.5 ThinkPHP 程序 调试 机 制 


ThinkPHP 内 荀 了 一 些 用 于 程序 调试 或 测试 的 类 库 ， 以 便 在 程序 出 现 寞 第 时 能 够 迅速 找 
到 问题 之 所 在 。 接 下 来 将 结合 示例 代码 ， 详 细 介 绍 系统 内 置 的 几 种 易 用 和 全 面 的 调试 方式 。 


8.5.1 开启 调试 功能 


调试 功能 是 ThinkPHP 内 置 的 一 项 重要 的 代码 调试 机 制 。 由 于 MVC 框架 是 由 众多 PHP 
类 库 构 造 的 ， 它 改变 了 许多 传统 PHP 开发 的 模式 ， 例 如 数据 库 操 作 、 文 件 包 含 、 代 码 执行 
流程 等 ， 在 开发 过 程 中 要 对 代码 进行 调试 ， 使 用 传统 的 PHP 来 调试 ， 只 能 调试 当前 PHP 代 
人 码 ， 但 不 能 调试 由 框架 引起 的 寞 党 。 所 以 几乎 所 有 主流 的 MVC HEREDERA E e Ae ENL 
制 ， 即 代码 调试 机 制 。ThinkPHP 本 映 内 置 有 代 人 码 调 试 功能 ， 开 发 人 员 只 需要 在 入 口 文 件 中 
开局 即 可 。 为 了 方便 开发 者 了 解 每 个 页 面 的 运行 情况 ， 系 统 还 提供 了 Trace 信息 显示 功能 。 
下 面 分 别 介绍 。 

1. 调试 模式 

一 旦 开启 调试 模式 ， 代 码 的 执行 步 又 将 会 发 生 改变 ， 这 种 改变 是 利于 代码 调试 的 ， 但 对 
运行 结果 没有 影 啊 。 要 局 动 代 人 码 调 试 功能 ， 设 置 常量 APP_DEBUG 为 mue 即 可 ， 如 以 下 代 
公所 示 。 


<?php 

em HTN HEET 
define (YARE BATH", "I/Nnome/A nw.) 

define ("APP_NAME", "Home"); 

define ("APP_DEBUG", true); 

require_once (THINK_PATH."ThinkPHP.php"); 


DEE TER, KRAER ERREN, mE 8-8 
所 示 。 

这 些 异 常 信息 根据 异常 的 情况 而 有 所 不 同 ， 并 且 是 可 以 进行 定制 显示 的 。 但 需要 注意 的 
是 调试 模式 只 有 异常 时 才 会 抛 出 调试 信息 。 
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系统 发 生 错 误 


车 可 以 选择 [ AE HE 或 者 [ 回 细 首页 ] 


ESO HLE DAD sampp-win32-1.7.4-VCB\mampp\htdocs\tpy demo bänkPH PN ba Behavior 
\LocationTemplateBehaviorclass php LNE: 23 


[17-10-71 150810] MADownloadswampp-wini?-L74A-VOpwamppihidocstpi demo Think PHALN Behavior 
\Location TempateBehaviordass php (23) Location TempmateBehavior-> parse Tempatefilet 
[17-10-271 150810] DADownloads\rampp- wind2- LA- VOBmampp hid pa demo ThinkPHP Comman 


Morpmonpbp Hi0 Location Temmatebehavior-=runmi 


2. Trace 异常 信息 

Trace 异常 信息 默认 是 关闭 的 ， 它 是 前 面 介 绍 的 异常 数据 显示 的 子 类 。Trace 信息 是 可 定 
制 的 ， 可 以 在 任何 运行 阶段 抛 出 ， 事 实 上 Trace 可 以 在 任何 动作 视图 上 随时 显示 。 

要 开局 Trace 页 面 信 息 ， 只 需要 在 配置 文件 中 设置 SHOW_PAGE_TRACE 为 true 即 可 。 
需要 注意 的 是 Trace 信息 需要 结合 模板 使 用 ， 也 就 是 说 动作 中 必须 使 用 $thin->display0 输 出 ， 


如 图 8-9 所 示 。 


mM Trace a 


Ab T Fe 
ST 


Ski : HTTR/1.1 GET 

运行 信息 : Process: 0.0080s ( Load:0.0004s Init:0.0013s Exec:0.0021s Template:0.0043s } | 
UseMem-42 kb | LoadFile:5 | GallFun:45,1671 

SRD ` wg3joranbebbsq3i7dohhftimad 

日 志 记 录 : 无 日 志 记 录 

WEF: 5 

[0] => D:\Downlbadswampp-win32-1.7.4-WVC6wamppihidocsitp3_demoindex. php 

[1] => D\Downlbadswampp-win32-1.7.4-VC6wamppihidocsiip3_demoiThinkPHPYThink PHP. php 
[四 => DADOwnloadswampp-win32-1.7.4VC6wamppuhnklocstdp3 demowhomeRuntimev-runtime.php 
[3] => DiDownloadswampp- -Wind2-1.7.4-VC6wampp\hidocsdtps demohoma\Lib\Action 
UndexAction class php 

Wi => DDownloadswampp -win32-1.7.4-vCewamppihtdocstp3 dermownomeRuntimnevCache 
ddDd46431aa157c89f3a1c30cd26be003b.php 


图 8-9 Trace 页 面 信 息 


Trace 信息 中 详细 地 列 出 了 当前 页 面 所 运行 的 时 间 、 状 态 、 内 存 占用 、 会 话 ID、 请 求 协 
议 以 及 页 面 所 有 调用 的 类 库 等 ， 这 对 于 代码 的 调试 及 优化 是 非常 有 用 的 。 

对 于 Trace 的 运行 时 间 、 状 态 、 内 存 占用 等 信息 ， 是 可 以 单独 提取 出 来 的 。 如 果 结 合 前 
面 介 绍 的 基于 消息 队列 的 日 六 系统 ， 那 么 还 可 以 将 运行 超时 的 页 面 推送 到 消息 队列 中 ， 方 便 


开 及 人 员 碍 看 并 优化 。 如 以 下 配置 文件 代码 所 示 。 


'SHOW_PAGE_TRACE' =>false, 

'SHOW_RUN_TIME '=>true, // 运行 时 间 显 示 
'SHOW_ADV_TIME'=>true, // 显示 详细 的 运行 时 间 
'SHOW_DB_TIMES'=>true, // ERA EA WAARA 
'SHOW_CACHE_TIMES'=>true, // 显示 缓存 操作 次 数 
'SHOW_USE_MEM'=>true, // 显示 内 存 开 销 


'SHOW_LOAD_FILE' =>true, // 显示 加 载 文件 数 
'SHOW_FUN_TIMES'=>true , // 显示 函数 调用 次 数 
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上 述 配 置信 息 将 会 显示 当前 动作 运行 时 间 、 数 据 库 操作 、 内 存 占用 等 信息 ， 读 者 可 以 根据 
需要 选择 开启 。 最 终 的 结果 将 会 在 页 面 底 部 呈现 〈 不 管 页 面 是 否 存在 异常 )， 如 图 8-10 所 示 。 


KREIS Sgr 订阅 


图 8-10 页 面 运行 状态 


8.5.2 ”代码 编译 概念 


PHP 本 喘 是 一 种 脚本 语言 ， 因 此 没有 编译 的 概念 。 这 里 所 说 的 代 但 编 详 是 指 ThinkPHP 
框架 提供 的 燃 库 缓存 功 能。 系统 为 了 提高 性 能 和 安全 性 ， 在 项 目 第 一 次 运行 时 会 将 所 需要 的 
系统 类 库 、 函 数 、 扩 展 、 配 置 文件 数据 等 载 入 到 内 存 ， 然 后 保存 到 Runtime/~runtime.php X 
件 中 ， 这 个 过 程 上 只 需要 一 次 ， 下 次 再 运行 项 目 时 ， 系 统 将 会 直接 谈 取 ~runtime.php 文件 中 的 
代码 缓存 ， 不 再 重复 调用 ， 这 样 驶 很 好 地 提高 了 性 能 。 同 时 ， 由 于 ~runtime.php 保存 的 是 最 
终 代 人 码 缓存， 所 以 在 排 答 错误 时 可 以 直接 奋 看 ~runtime.php 文件 中 的 绥 存 代码 ， 方 便 查 找 最 
终 的 问题 所 在 。 

由 于 代码 编译 会 将 框 避 中 的 核心 代 合 和 扩展 类 库 等 必需 的 代 但 绥 存 到 一 个 文件 中 ， 这 束 
意味 看 如 果 项 目 本 喘 没 有 发 生 配置 和 代码 变动 ， 开 发 人 员 甚 至 可 以 使 用 ~runtime.php 文件 代 
巷 框 架 入 口 文 件 。 也 就 是 说 项 目 不 再 需要 引入 ThinkPHP， 而 只 需要 引入 ~runtime.php 文件 即 
可 《包括 项 目的 配置 文件 、 目 定 函数 文 件 、 项 目 扩展 类 库 等 都 可 以 删除 )， 这 对 提高 网 站 的 
安全 性 是 有 帮助 的 。 

项 目 预 编译 文件 默认 存放 于 项 目 Runtime 根 目 录 下 ， 开 发 人 员 可 以 通过 修改 
RUNTIME_PATH 入 口 文件 配置 项 改变 ~runtime.php 存放 目录 。 此 外 ， 还 可 以 修改 入 口 文件 
RUNTIME_FILE 配置 项 改变 预 编 详 文件 名 称 ， 如 以 下 代码 所 示 。 


<?php 
define ("THINK_ PATH","./ThinkPHP/"); 
define ("APP_ PATH","./home/"); 


define ("APP NAME", "Home"); 
define ("APP DEBUG", false); 
define ('RUNTIME_PATH' ,APP_PATH. 'temp/'); // 预 编译 文件 文件 目录 
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define ('RUNTIME_FILE', RUNTIME_PATH.' runtime_cache.php');// 预 编 译文 件 名 
require_once (THINK_PATH."ThinkPHP.php"); 
?> 


卉 党 信息 是 调试 模式 中 最 和 直观 的 数据 ， 这 些 信息 的 处 理 是 由 系统 基 类 ThinkException 类 来 
完成 的 。 开 友人 员 可 以 对 卉 第 信息 进行 定义 ， 包 括 并 第 信 息 显 示 项 定义 和 模板 定义 。 这 些 操作 
虽然 对 程序 调试 意义 不 大 ， 但 项 目的 文件 组 件 和 管理 却 有 一 定 的 意义 。 为 了 方便 在 开发 中 进行 
寞 单 处 理 ， 系 统 还 提供 了 throw_exception 图 数 用 于 处 理 异 利信 息 ， 该 函数 封 帮 了 所 有 对 
ThinkException 类 的 调用 操作 ， 并 且 不 受 APP_DEBUG 入 口 文件 配置 项 的 影响 。 如 果 在 生产 环 
卉 中 部 普 ， 不 需要 显示 这 些 敏 感 的 数据 ， 或 者 需要 统一 收集 这 些 开 笛 信 息 ， 系 统 还 文 持 使 用 一 
个 控制 器 动作 或 者 URL 作为 统一 处 理 异 音 的 页 面 。 接 下 来 将 详细 介绍 异 利信 息 的 定制 。 

1. ThinkException 类 

ThinkException 类 继承 于 PHP 内 置 的 Exception 异常 错误 处 理 类 ， 提 供 了 目 定 义 模板 和 目 定 
义 信息 的 功能 。 默 认 和 情况 下 ， 措 党 信息 的 显示 模板 路 人 径 为 ThinkPHP/Tpl/think_exception.tpl， 开 发 
人 员 可 以 通过 修改 EXCEPTION_TMPL_FILE 配置 项 修改 异常 信息 模板 。 如 以 下 代码 所 示 。 

'EXCEPTION_TMPL_FILE' => '/Public/exception.tpl', 


'SHOW_ERROR_MSG' =>true, 
TERROR MESSAGE = RT Rl 


通过 修改 EXCEPTION_TMPL _FILE 配置 项 ， 此 时 的 异常 模板 将 会 使 用 自 定义 的 模板 
/Public/exception.tp1， 人 代码 如 下 所 示 。 


<html> 
<head> 


<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 

<title> 错 误 信 息 </title> 

</head> 

<body> 

<div class="main"> 
<1Li> 异 常 文件 ，<?PhP echo $e['file'];?></li> 
<1Li> 异 常 发 生 行 数 : <?php echo $e['line'];?></li> 
<1Li> 异 常 信 息 : <?php echo $e['message'];?></li> 
<1Li> 异 常 详细 代码 : <?php echo $e['trace'];?></li> 

</div> 

<div> 

客户 端 环境 : <?php echo $_SERVER["HTTP_USER_AGENT"] ; ?> 

</div> 

</body> 

</html> 


如 上 述 代码 所 示 ， 由 于 ThinkException 类 继承 于 Exception， 所 以 可 以 使 用 关键 学 throw 
捕捉 寞 党 数据， 并 保存 到 全 局 数组 中 ， 开 发 人 员 可 以 手动 获取 数组 中 的 寞 党 信息 。 
ThinkException 类 对 Exception 类 捕捉 到 的 卉 第 数 据 进行 了 二 次 处 理 ， 不 需要 在 馆 辑 层面 直接 
输出 ， 只 和 需要 在 模板 中 直接 引用 即 可 。 这 也 束 意 味 看 在 模板 中 定义 的 PHP 代码 都 能 够 被 系 
统 执行 。 在 动作 逻辑 层面 ， 只 需要 使 用 快捷 函数 throw_exception 调用 模板 即 可 。 


<?php 


class IndexAction extends Action { 
public function hello(){ 
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Şuser=M ("User"); 


Şrows=$user—->where ("sid>0")->select(); 
ŞgetLastSsql=Ş$Şuser->getLastsql(); 
if (empty ($rows["id"])){ 
// 出 错 提示 
thzrow_excepPtion(" 查 询 出 错 ， 最 后 执行 SQL 语句 为 : " .$SgetLastSql); 
}else{ 
/ /逻辑 处 理 


} 
} 
} 


?> 


throw_exception 疯 数 只 是 封装 了 throw_exception 类 的 所 有 操作 ， 如 果 习 惯 传统 的 throw 
KEFALET, ThinkPHP 允许 开发 人 员 继 续 使 用 throw 关键 学， 如 以 下 代码 所 示 。 


throw new ThinkException ("代码 执行 出 错 ; ".$getLastsql); 

2. 使 用 URL 处 理 异 常 

如 果 项 目 己 经 部 署 到 生产 环境 中 ， 还 可 以 通过 配置 ERROR_PAGE 选项 指定 一 个 URL 
或 者 控制 器 动作 收集 异常 信息 。 例 如 当 数 据 不 存在 ， 或 者 下 载 文件 不 存在 时 系统 怠 跳 转 到 一 
个 页 面 ( 类 似 于 404 错误 页 面 )， 如 果 是 一 个 控制 器 动作 ， 还 可 以 在 动作 使 用 消息 队列 发 送 
邮件 给 管理 员 。ERROR_PAGE 配置 项 非常 简单 ， 如 以 下 代码 所 示 。 

ie 

一 旦 配置 了 ERROR_PAGE 选项 ， 系 统 会 忽略 EXCEPTION_TMPL _ FILE 选项 。 这 种 方式 
使 用 的 是 传统 的 跳 转 方式 ， 相 当 于 服务 器 中 的 404 异常 处 理 ， 这 对 于 访客 而 言 是 比较 友好 的 。 


8.5.4 性 能 调试 


Debug 是 一 个 系统 静态 扩展 类 ， 主 要 用 于 调试 程序 的 性 能 ， 为 优化 程序 运行 效率 提供 依 
据 。 例 如 显示 块 代码 内 存 占用 、 块 代码 执行 时 间 、 区 间 代 码 性 能 调用 等 。 此 外 系统 还 提供 了 
快捷 函数 G 用 于 简单 地 测试 代码 性 能 ， 下 面 分 别 介绍 。 

1. Debug 类 

Debug ZS ear KR, H TIARE. ARANT 8 IO un e ENA: 
mark, useTime, useMemory, getMemPeak. F mark 方法 为 Debug 类 的 核心 方法 ， 后 面 3 
个 方法 必须 以 mark 为 统计 依据 才能 收集 到 代码 执行 性 能 数据 ， 下 面 分 别 介绍 。 

(1) mark 

mark 是 Debug 调试 类 的 核心 方法 ， 该 方法 不 参与 数据 收集 ， 只 用 于 声明 调试 的 开始 与 
结束 标记 。 假 设 需要 对 一 个 页 面 进 行 性 能 调试 ， 那 么 首先 需要 在 测试 的 代码 之 前 加 入 标记 ， 
如 以 下 代码 所 示 。 


public function test (){ 
import ('ORG.Util.Debug'); 
Debug: :mark ('run'); 


ŞcommentObj=M ("Comments"); 
Şrows=$commentObj->select (); 
Debug: :mark ('end'); 


} 
如 上 述 代码 所 示 ， 由 于 Debug 是 一 个 系统 扩展 类 ， 所 以 需要 引入 扩展 。 关 于 扩展 的 详细 
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使 用 本 书 第 11 间 将 有 详细 介绍 。 在 此 读者 只 需要 理解 即 可 。mark 方法 只 需 传 入 一 个 参数 即 
可 ， 该 参数 是 一 个 可 目 定 义 的 字符 串 。 例 如 run 与 end， 表 示 代 但 调试 从 Debug::mark(run") 开 
D, HS Debug::mark(Cend) 结 束 。 这 对 标记 内 的 代码 性 能 数据 会 被 记录 到 $marker 数组 中 。 
如 果 要 对 一 个 完整 的 控制 器 动作 代码 进行 调试 ， 更 理想 的 方法 是 在 前 动作 和 后 动作 中 置 
入 mark 标记 ， 这 样 更 有 利于 代码 管理 ， 如 以 下 代码 所 示 。 
EEE RIE 
public function _before_test () { 


import ('ORG.Util.Debug'); 
Debug: :mark ('run'); 


} 
public function test () { 
ŞcommentObj=M ("Comments"); 
Şrows=$commentObj->select(); 


| 

ee 

publie function atter test()1 
Debug: :mark ('end'); 


} 
为 代码 添加 了 标记 后 ， 系 统 束 能 够 识别 调试 区 域 7 了 了。 其 他 的 3 种 方法 分 别 用 于 获取 代码 
运行 时 间 、 代 人 码 运 行 占用 内 存 及 代码 占 用 内 存 的 峰值 。 
(2) useTime 
useTime 用 于 收集 标记 区 代码 的 运行 时 间 ， 访 方法 共 文 持 3 个 参数 ， 表 现形 式 为 
useTime($start,$end,$decimals = 6)。 其 中 start 及 end 参数 即 为 mark 方法 中 对 应 的 开始 标记 及 
结束 标记 ; decimals 表示 时 间 统 计 精 度 。 代 码 如 下 所 示 。 
ea EE 
public function _before_test () { 


import ('ORG.Util.Debug'); 


Debug: :mark ('run'); 


} 

publie function test (|) i 
ŞcommentObj=M ("Comments"); 
Şrows=$commentObj->select (); 


} 
/ {test 后 动作 
public function _after_test(){ 
Debug: :mark ('end'); 
echo "运行 所 需 时 间 : " .Debug: :useTime ('run', end"). "br: 


} 
如 果 动 作 中 使 用 了 缓存 ，useTime 能 够 真实 地 收集 到 绥 存 后 的 运行 时 间 ， 这 对 于 绥 存 优 
化 是 非常 有 用 的 。 
(3) useMemory 
useMemory 方法 用 于 收集 区 间 人 代码 所 占用 的 系统 内 存 。 设 方法 共 文 持 两 个 参数 ， 表 现形 
式 为 useMemory($start,$end)。start 及 end 参数 分 别 代码 mark 方法 中 对 应 的 开始 标记 及 结束 
标记 。useMemory 的 实际 使 用 如 以 下 代码 所 示 。 


ER 有 人 


public function _oOetore Eese() 
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import ('ORG.Util.Debug'); 
Debug: :mark ('run'); 
} 
public function test () { 
ŞcommentObj=M ("Comments"); 
Şrows=$commentObj->select (); 


} 
//test 后 动作 
public function _after_test(){ 
Debug: :mark ('end'); 
echo "页 面 占用 内 存 : ".Debug: :useMemory ('run', 'end')."kb"; 


} 

在 实际 应 用 开发 中 ， 可 以 根据 需要 换算 useMemory 结果 值 。 

(4) getMemPeak 

useMemory 得 到 的 结 末 是 区 间 代 人 码 最低 占 用 率 值 与 最 高 占用 率 值 之 间 的 平均 值 。 而 
getMemPeak 方法 用 于 获取 区 间 代 但 内 存 占 用 率 的 最 高 值 ， 所 以 _ getMemPeak 方法 更 具有 参 
ZAX. getMemPeak 方法 与 useMemory 方法 无 论 参 数 还 是 什么 都 是 一 样 的 ， 在 此 吏 不 再 重 
复 相关 代码 演示 。 需 要 说 明 的 是 ， 无 论 是 useMemory 还 是 getMemPeak， 使 用 的 前 提 都 是 
Web 服务 器 安装 并 开启 了 memory_get_usage 子 数 ， 厂 则 将 获取 不 到 任何 数据 。 

2. G 快捷 函数 

G 快捷 函数 能 够 丛 单 快捷 地 获取 区 间 代 码 性 能 数据 。 相 比 Debug X, G 快捷 函数 不 需要 
引 手 动 引 入 扩展 类 ， 只 需要 定义 开始 标记 即 可 ， 结 束 标 记 是 可 选 鸭 。G 快捷 函数 文 持 3 个 参 
数 ， 表 现形 式 为 G($start,$end=",$dec=4)。 其 中 start 表示 需要 传 入 开始 标记 ; end 是 可 选 参 
数 ， 如 果 为 空 即 表 示 以 当前 动作 最 后 一 行 代 人 码 为 基准 ; dec 是 可 选 参数 ， 该 参数 值 为 数字 关 
型 ， 用 于 设 定 统计 精度 。 使 用 G 函数 比较 人 简单， 如 以 下 代码 所 示 。 


//test 前 动作 


publice function _oboetore tese()l 


G('run'); 
} 
public function Test|) i 
ŞcommentObj=M ("Comments"); 
Şrows=$commentObj->select (); 


} 
人 
EC function after tesct()1 


echo "代码 执行 时 间 : ".G('run', 'end').' 秒 '; 


} 
相对 Debug Km A> G 函数 收集 的 信息 比较 简单 ， 收 集结 末 只 有 代码 运行 时 间 。 但 大 多 
数 情况 下 ， 根 据 代 码 执行 的 时 间 残 可 以 得 到 代码 的 运行 情况 和 效率 。 


8.6 小结 


本 章 深 入 地 介绍 了 ThinkPHP 开发 MVC 应 用 程序 中 安全 与 调试 的 方方面面 。 首 先 一 开 
台 束 使 用 了 形象 、 易 全 的 流程 图 对 系统 内 首 的 安全 机 制 进 行 了 描述 ， 然 后 结合 代码 重点 讲解 
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了 表单 令 脾 、 数 据 验 证 等 实用 技术 。 

接 看 介绍 了 系统 内 症 的 日 六 处 理 机 制 。 养 成 民 好 的 目 定 义 日 六 记录 ， 将 能 够 给 项 目的 后 
期 维护 市 来 极 大 的 便利 ， 所 以 笔者 重点 介绍 了 目 定 义 日 六 的 使 用 。 性 能 与 日 六 系统 是 相 辅 相 
成 的 ， 如 果 面 临 超大 的 并 发 ， 传 统 的 日 忘记 录 方 式 将 难以 应 对 ， 所 以 本 章 还 讲述 了 消息 队列 
机 制 ， 它 使 用 卉 步 处 理 机 制 ， 能 够 解决 大 并 发 融 来 的 各 种 问题 。 最 后 介绍 了 系统 内 置 的 代码 
性 能 调试 兴 ， 使 得 初学 者 也 能 够 对 代码 性 能 进行 有 效 测试 。 

理解 本 章 内 容 ， 对 构建 安全 及 稳定 的 MVC 应 用 是 非常 有 帮助 的 ， 读 者 应 该 融会 贯通 。 下 
一 革 将 继续 介绍 ThinkPHP 内 置 的 各 种 功能 强大 的 类 库 ， 帮 助 读者 有 效 提 高 开发 效率 。 
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通常 情况 下 我 们 可 以 通过 自 定义 函数 库 为 项 目 提供 更 多 功能 ， 并 将 应 用 开发 过 程 中 多 
次 重复 的 代码 封装 为 函数 。ThinkPHP 功能 库 能 够 简化 开发 人 员 编 写 自 定义 函数 的 步骤 ， 将 
更 多 编程 工作 放 到 远 辑 处 理 上 。 当 然 系统 内 置 的 功能 库 并 不 能 代 赫 自 定义 函数 库 ， 但 是 一 
些 常 用 的 功能 可 以 首先 在 功能 库 中 查找 ， 如 果 存 在 则 直接 使 用 ; 如 果 没 有 则 再 使 用 自 定义 
马 数 编写 。 

在 实际 应 用 开发 中 ， 更 有 效率 的 原则 是 以 系统 内 置 的 功能 库 为 基础 ， 然 后 在 自 定义 函数 
库 中 进行 封 玲 ， 使 得 功能 库 的 功能 和 荔 用 性 得 到 最 大 提升 。 这 里 所 说 的 功能 库 是 有 一 定 的 局 
限 性 的 ， 事 实 上 自 定义 函数 库 也 是 基于 系统 基础 类 库 的 ， 所 以 在 自 定 义 函 数 中 直接 使 用 系统 
类 库 、 配 置信 息 、CURD 操作 等 都 是 被 允许 的 。 

本 章 将 会 对 数据 处 理 、 函 数 扩 展 、 多 语言 、Session 等 功能 库 进 行 详 细 介 绍 ， 帮 助 读者 
在 实际 应 用 开发 中 提高 效率 。 


学 习 目 标 


了 解数 据 处 理 引 擎 概念 。 

掌握 SAX 引擎 的 使 用 。 

掌握 Dom 的 使 用 。 

了 解 Json 及 Jsonp。 

掌握 SimpleXML 扩展 的 使 用 。 
了 解 ThinkPHP 扩展 函数 库 。 
了 解 ThinkPHP 扩展 类 库 。 
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9.1 数据 处 理 


传统 的 网 站 开发 中 ， 一 般 开发 人 员 只 需要 处 理 HTML 或 者 纯 文本 输出 即 可 ， 但 是 由 于 
互联 网 的 迅猛 发 展 ， 一 个 成 熟 的 网 站 通常 都 不 是 独 目 构 成 的 ， 而 是 由 网 站 、API、 提 面 客户 
端 、 手 机 客户 端 、UC (网 站 实时 聊天 工具 )、 统 计 分 析 系 统 等 共同 构成 。 要 将 这 些 功 能 模块 
或 系统 整合 到 一 起 ， 使 用 纯粹 的 HTML 或 者 文本 来 通信 ， 显 然 不 可 行 。 在 主流 网 站 开发 技 
术 中 《例如 CH Java) 都 有 各 目的 路 应 用 通信 技术 ， 其 中 最 通用 的 驶 是 Json 及 XML. PHP 
本 喘 内 置 有 处 理 Json 及 XML 的 类 库 ， 但 使 用 复杂 ，ThinkPHP XHHT "Nä: uch, (dr 
得 该 类 库 更 加 适合 在 MVC 中 使 用 。 


9.1.1 XML 引擎 


XML 是 一 种 标记 性 语言 ， 由 GML (一 种 工业 平面 印刷 描绘 语言 ) 派生 而 来 。 由 于 它 的 
扩展 性 好 、 可 编程 性 高 、 数 据 传 输 量 大 等 特点 ， 所 以 很 早 束 被 用 于 Web 数据 交互 。 早 在 
2004 年 ， 微 软 发 布 IE6 浏览 器 时 吏 能 够 文 持 简单 的 XML 通信 ， 由 于 XML 国际 化 标准 的 设 
定 ， 近 年 来 各 类 编程 语言 几乎 都 对 XML 提供 了 全 方位 的 文 持 ， 有 些 编程 技术 甚 全 单独 内 置 
XML 解释 引擎 (例如 Object-C 编程 )， 足 见 XML 的 活力 及 优势 。 

早期 的 PHP 版 本 (4.x) 对 XML DIr ten, (OCI PHP 已 对 XML 提供 了 完 
善 的 文 持 “〈 无 论 是 解释 还 是 生成 )。PHP 5.x 对 XML 的 处 理 主要 有 4 种 方式 : SAX (Simple 
Api for XML) 引擎 、Dom (Document Object Model) 解释 句 、SimpleXML 扩展 、XML Reader 
扩展 。ThinkPHP AEH XML 处 理 技术 也 是 基于 上 述 方式 的 。 下 和 面 首 先 对 PHP 原生 的 XML 
处 理 机 制 进行 简单 讲解 ， 然 后 结合 ThinkPHP 框架 讲解 在 MVC 中 处 理 XML 文档 。 

1. SAX 

SAX 引 敬 是 一 球 轻 量 级 的 XML 处理 引擎 ， 它 的 执行 过 程 是 由 上 而 下 顺序 执行 的 。SAX 
在 处 理 XML 时 使 用 的 是 异步 事件 驱动 机 制 ， 这 能 够 在 快速 反应 的 同时 及 时 地 捕捉 到 XML 
中 的 节点 信息 ， 由 于 SAX 是 单 问 无 缓冲 的 ， 所 以 SAX 非常 适用 于 手机 终端 等 内 存 受 限 设备 
上 AOS, Android 等 默认 使 用 SAX 来 解释 XML). 

PHP 作为 一 门 网 站 编程 语言 ， 在 讲求 运行 效率 的 网 站 应 用 需求 上 ， 同 样 也 默认 使 用 内 置 
的 SAX 解释 引擎 处 理 XML。 开 发 人 员 在 安装 完 PHP 5.X 之 后 就 能 够 使 用 SAX 开发 XML 网 
站 了 。 下 面 将 通过 一 个 简单 的 示例 讲解 SAX 解释 XML 的 全 过 程 。 

假设 网 络 上 有 一 个 XML 接口 ， 现 在 需要 使 用 PHP 来 解释 该 文档 中 的 数据 ， 以 便 进 行 开 
Ro XML 文档 数据 如 以 下 代码 所 示 。 


<?xml version="1.0" encoding="utf-8"?> 
<result> 


<row> 
<id>1</id> 
<date>1351048596</date> 
<title>PHP 比 你 想象 的 好 得 多 </title> 
<content> 在 说 最 近 PHP 社区 取得 的 惊人 成 束 之 前 ， 我 们 先 来 看 看 一 些 有 趣 的 数学 : PHP 被 77.9% 的 服务 端 编 
程 语言 已 知 的 网 站 使 用 。Wordpress 被 全 世界 16.6% 的 网 站 使 用 。 使 用 率 最 高 的 三 个 CMS 建站 系统 是 : 第 一 的 
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Wordpress 份额 为 54 .3%， 第 二 的 Joomla 份额 为 9.2%， 第 三 的 Drupal 份额 为 6.8$。 这 三 个 产品 都 是 用 PH 5 
的 。 
</content> 
</row> 
<row> 
<id>2</id> 
<date>1351048677</date> 
<title>PHP 5.4 内 置 Web 服务 器 </title> 
<content>PHP 是 一 种 脚本 语言 ， 它 需要 PHP 解释 器 来 分 析 运 行 PHP 文件 。 当 把 PHP 做 为 CGI 服务 web 请求 
时 ， 它 需要 被 柳 入 到 某 种 Web 服务 器 里 ， 最 常见 的 是 集成 到 Apache 或 IIS 里 ， 这 就 是 说 ， 在 使 用 PHP 前 ， 你 需要 安装 
Apache 或 IIS， 并 且 正 确 的 配置 它们 和 PHP 集成 的 参数 。 虽 然 这 种 配置 已 经 很 规范 ， 文 档 非 党 丰富 ， 但 我 们 还 是 经 党 
在 安装 Apache 和 PHP 集成 时 遇 到 问题 ， 而 且 ， 有 时 候 我 们 只 想 测 试 一 个 简单 的 PHP 特征 ， 不 想 就 为 此 安装 、 局 动 
Apache 服务 。 
</content> 
</row> 
</result> 


上 述 XML RERA ARAIZ rt 2e Or RE ER. E RKE 
用 xml_parser_create 函数 创建 XML 对 象 ， 并 使 用 xml_parse 来 提取 文档 中 的 节点 信息 。 这 两 
个 函数 是 SAX 引擎 的 核心 函数 ， 使 用 时 与 普通 的 PHP 函数 并 无 区 别 。 为 了 方便 代码 管理 ， 
这 里 将 SAX 创建 XML 数据 对 象 及 解释 XML 数据 全 过 程 封装 成 一 个 功能 类 ， 并 命名 为 
SaxXmlClass， 代 码 如 下 所 示 。 


<?php 
class SaxXmlClass { 


private $parser; 

private $i = 0; 

private S$search result = array (); 
private $row = array (); 

public $data = array (); 

private Snow tag; 


ek E EE EE (WIDY, VELASSIDY, De EE NEE DE OM 
Et EE EE EE E E EEN GEO DEE 
/ /构造 XML 数据 对 象 
function construct() { 
StaLs=>parser = xml parser create (UT 8!)) 
xml ser objecte ( Srais=->parger, Srhnis )? 
xml_set_element_handler ( S$this->parser, "tag_open", "tag_close" ); 
ml set Characcer daca_nhanciler ( Sthls=->parser, Vedarcat ) y 


} 

// 使 用 xml_parse 解释 数据 

public function parse ($data) { 
sml parse ( Sthis->parcser, Sarca rs 

} 

/7 返回 XML 节点 

protected function tag_open ($parser, Stag, Sattributes) { 
Şthis->now_tag = $tag; 


if ($tag == 'RESULT') { 
$this->search_result = Ş$attributes; 
} 
if ($tag == 'ROW') { 
ŞSthis->row [Senlis >1) = farci ribures? 
} 
} 
// 生 成 数据 对 象 
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protected function cdata ($parser, Scdata) { 


ii (in array ( Şthis=>now_ tacy Schis=>tags )) 1 
stagname = strtolower ( Sthis->now tag );} 
Sthis->qata [Ş$this->i] [Ş$tagname] = $cdata; 

} 

} 

/ /查找 XML 标记 结束 

protected function tag_ close(S$parser, Stag) { 
SrtChise-zpow cac = Di: 
十 全 (Stag == ROW”) 4 


Srhbis-zi kk: 


?> 


最 后 只 需要 在 使 用 页 面 中 实例 化 SaxXmlClass 类 ， 并 调用 data St Hu, Gebei 
最 终生 成 的 数据 对 象 ， 代 码 如 下 所 示 。 


<?php 
require_once ("Lib/saxXmlClass.php"); 


$xml = file_get_contents ("http://localhost/web/news.xm1l"); 
şxml_parser = new SaxXmlClass(); 

şxml_parser->parse ($xml); 

sdata=sxml parser->data; 

//var_dump ($data); 

foreach ($data as $key=>Şvalue){ 


SO 
站 


} 


将 代码 保存 ， 和 直接 访问 页 面 将 会 看 到 运行 结果 。 这 些 狐 闻 数据 不 是 来 日 于 数据 库 ， 也 不 
是 来 目 于 纯 文本 ， 而 是 来 目 于 具备 路 站 通信 能 力 的 XML。 

2. DOM 

DOM 全 称 为 Document Object Model (NERZ), MEREM UA H DOM 是 专 
门 用 于 处 理 文 档 数 据 的 。 事 实 也 如 此 ，DOM 能 够 处 理 多 种 文档， 包括 常见 的 SGML, 
HMTL、XHTML、XML、RSS 等 。PHP 内 置 的 DOM HFA HTML, XML 等 是 非常 合适 
的 ， 也 是 非常 高 效 的 。 

与 SAX 不 同 ，DOM 处 理 XML 时 首先 将 XML 加 载 到 服务 器 内 存 ， 然 后 再 进行 解释 ， 
这 种 解释 方式 是 不 需要 按照 顺序 执行 的 ， 最 开始 载 入 的 和 点 最 终 的 解释 顺序 未 必 吏 排 在 最 前 
面 。 由 于 DOM 首先 将 XML 载 入 内 存 ， 这 就 意味 着 在 引擎 还 没 最 终 和 输出 结果 前 ， 开 发 人 员 可 
以 对 内 存 中 的 数据 进行 增删 、 改 ， 这 也 是 DOM 最 为 强大 的 功能 。 同 时 ， 由 于 DOM 是 先 载 
入 后 解释 的 (SAX 是 同时 载 入 同时 解释 )， 使 得 数据 能 够 疏 缓存 ， 所 以 在 解释 大 型 的 XML 数 
据 时 DOM 性 能 更 加 出 色 ， 接 下 来 将 继续 以 前面 的 例子 为 基础 ， 详 细 介 绍 DOM 的 使 用 。 

(1) DOM 查询 XML 

AEH ASX 引擎 能 够 方便 地 对 XML 数据 进行 合 询 ， 接 下 来 将 使 用 DOM 进行 同样 的 
查询 操作 ， 帮 助 读者 加 深 对 DOM 与 SAX 的 认识 。PHP 5.x 对 DOM 的 文 持 已 经 非常 完善 ， 
开发 人 员 不 需要 再 手动 安装 插件 ， 只 和 需要 实例 化 DOMDocument 基础 类 即 可 。 访 关 提 供 了 所 
有 对 文档 进行 操作 所 需要 的 方法 和 属性 ， 其 中 用 于 查询 XML 文档 的 方法 主要 为 
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documentElement、getelementsByTasName、childNodes。 
其 中 documentElement 用 于 得 到 XML 文档 ` Ehr? EE 返回 市 有 指 
定 标 签名 的 对 象 集 合 ，childNodes 返回 所 有 子 节 点 (最 内 层 的 节点 )。 熟 悉 JavaScript 的 读者 ， 
相信 对 DOM 并 不 会 感到 卫生 。 事 实 上 ，PHP 内 置 的 DOM 5| Ve JavaScript 中 的 DOM 引擎 
使 用 方式 上 都 是 相似 的 。 下 面 将 结合 代 但 演示 DOM 查询 news.xml 文档 的 全 过 程 。 
首先 创建 一 个 功能 类 ， 并 命名 为 DomNewsXML。 该 类 于 用 创建 解释 占 对 象 ， 并 是 完成 
数据 的 合 询 ， 如 以 下 代码 所 示 。 


<?php 
class DomNewsXML extends DOMDocument { 


private Groot: 


/ / RI RR är 
public function _ construct () { 
parent::_ construct (); 
Şthis->load ( "news.xml" ); 
} 
// 解 释 XML 
public function show message () { 
Şroot = Şthis->documentElement; 
$xpath = new DOMXPath ( $this ); 
Şnode_record = Ş$this->getelementsByTagName ( "row" ); 


snode_record length = $node_record->length; 
sdatas=array ();}; 

/ E] row "DO 

ror($i = 0; 81 < node recor =>lengehn; $i Pr) 4 


2 

foreach ( $node_record->item ( $i )->childNodes as Sarticles ) { 
$data [$k] = $articles->textContent; 
Sk EE 

} 

/ /包装 数 据 


Salsi eels 
sd[sil]["time"]=$datal3]; 
sdatas=$d; 

} 

/ /返回 数据 


return S$datas; 


SE 


将 上 述 代 人 码 保存 ， 在 使 用 文件 中 只 需要 调用 how message 方法 即 可 得 到 news.xml 文件 
中 的 数据 ， 调 用 文件 代码 如 下 所 示 。 


<?php 

include_once ("DomNewsXML .class .php");} 
shawkXML = new DomNewsXML () ; 
Şdata=$hawkXML->show_message (il: 
//var_dump ($data); 

foreach ($data as $key=>Şvalue){ 


echo <- mos KEE D EA a ee anae S a dae E EE 
| 
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} 


2S 


该 示例 的 运行 结果 和 前 面 使 用 SAX 的 结果 是 一 致 的 。 读 者 可 以 在 此 基础 上 继续 完善 ， 
实现 更 复杂 的 XML 文档 查询 。 

(2) DOM 增加 XML 元 素 

DOM 最 强大 的 功能 之 一 就 是 可 以 对 文档 模型 进行 CURD 操作 。 其 中 对 XML 增加 文档 
元 素 ， 只 需要 调用 DOMDocument 基 类 类 中 的 appendChild 方法 即 可 。 这 里 继续 以 
DomNewsXML.class.php 文件 为 例 ， 在 该 类 中 添加 add_news 方法 ， 议 方法 用 于 实现 添加 新 
闻 ， 代 人 码 如 下 所 示 。 


private $root; 


/ /构造 函数 

public function _ construct() { 
parents EE EE 
SE E En 

} 

// 解 释 XML 

人 

// 增 加 新 闻 

public function adqd news ($date, $title, $content) { 
/ /初始 化 文档 对 象 
Soot = Şthis->documentElement; 


// 定 义 新 闻 id 子 节点 
Se 0 


$s id = $this->createElement ( “id™ A: 

$id = $this->createTextNode ( iconv ( "UTF-8", "UTF-8", $id ) ) ，; 
Ş_id->appendChild ( $id ); 

/ /定义 新 闻 日 期 了 子 市 点 

$s date = $this->createElement ( "date™ ) ， 

$date = $this->createTextNode ( iconv ( "UTF-8", "UTF-8", $date ) ); 


ş_date->appendChild ( $date ); 

/ /定义 狐 闻 标题 子 市 点 

$ cicle = SENLS ->Onreacenleanene ( VYeicle” es 

stitle = $this->createTextNode ( iconv ( "UTF-8", "UTF-8", Stitje ) ); 

S$ _ tlicle EE 

// 定 义 新 闻 内 容 子 节点 

$ Content = Ş$this->createElement ( "content" ji: 

Seonmenc = SEnis— >ereacerexcNeoe ( Leony ( “UT = 8, VUIF $content ) )} 
Ş_content->appendChild ( $content ); 

// 创 建 row 节点 

SNode_ Record = Sthis->createElement ( "row" ) ，; 

7 DA 
ŞNode_Record->appendChild 
ŞNode_Record->appendChild 
ŞNode_Record->appendChild 
ŞNode_Record->appendChild 
// 将 子 节 点 添加 到 row 节点 下 
ŞRoot->appendChild ( $Node_Record 1: 
Eed 


id ); 
carce )e 
Content ) p 


( S_ 
( $ 
( S_ 
人 


$this->save ( "news.xml" ji: 
SE EE 

} cacca ( mrrorbxecepclon ) | 
throw $e->getMessage (); 
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} 
} 


Dm TTC Dirr, "Dr news.xml 文档 添加 row 子 节 点 ， 束 相当 于 添加 新 闻 数 据 。 整 个 流 
程 共 分 4 步骤 : 首先 使 用 documentElement 方法 根据 构造 函数 中 载 入 的 XML 文件 创建 文档 
对 象 ， 然 后 定义 数据 节点 ， 并 使 用 appendChild 方法 添加 到 文档 对 象 中 ;， 接 着 将 所 有 节点 添 
加 到 row 父 节 点 下 ; 前 面 的 步骤 都 是 在 内 存 中 完成 的 ， 最 后 还 需要 使 用 save 方法 写 入 到 
news.xml] 文件 中 ( 即 保 存 到 便 稚 中 )。 

需要 注意 的 是 ， 创 建文 档 对 象 节 点 的 过 程 中 ，PHP 中 的 DOM 引擎 对 中 文 汉字 的 文 持 存 
在 BUG， 开 发 人 员 需 要 手动 将 汉字 显 式 地 转换 为 UTF8 编码 〈 束 算 当 前 文件 为 UTES 编码 也 
需要 转换 )。 

最 后 在 使 用 文件 中 和 直接 调用 add_news 方法 并 传 入 狐 闻 数据 参数 即 可 ， 如 以 下 代码 所 示 。 


<?php 


include_once ("DomNewsXML.class.php"),，; 
$DomAddNews=new DomNewsXML () ; 
$obj=$DomAddNews->add_news (time (), 
"为 您 介绍 5 个 PHP 安全 措施 "， 
"多 年 来 ，PHP 一 直 是 一 个 稳定 的 、 廉 价 的 运行 基于 web 应 用 程序 的 平台 。 
像 大 多 数 基于 web 的 平台 一 样 ，PHP 也 是 容易 受到 外 部 攻击 的 。") ; 
(oD 

echo "数据 添加 成 功 "; 
EE 


echo "数据 添加 失败 "; 


} 


(3) DOM 删除 XML 元 素 

使 用 DOM 删除 XML 元 素 非 党 和 价 单 ， 只 需要 调用 DOMDocument 其 类 中 的 removeChild 
方法 即 可 。 下 面 继 续 以 DomNewsXML.class.php 文件 为 例 ， 在 该 类 中 添加 delete_news Jr 
法 ， 实 现 新 闻 数 据 的 删除 ， 代 码 如 下 上 所 示 。 


private $root; 


/ /构造 函数 

Buolie function —_ Construct) 1 
parent::_ construct (); 
Sthails=>loac) ( Wmews mL" ) 7 

} 

// 解 释 XML 

CR 

/ /删除 新 闻 

public function delete news ($id) { 
ŞRoot = Şthis->documentElement; 


Şxpath = new DOMXPath ( $this ); 
ŞNode_Record = $xpath->query ( "//row[id='$id']" )}; 


$Root->removeChild ( $Node_Record->item ( 0 ) )，; 
tryd 
$this->save ( "news.xml" ji: 


LOLUEN erte, 
} cacca ( mrrorbxcepcion Se ) í 
throw $e->getMessage (); 


} 
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} 

通过 前 面 的 例子 ， 可 以 看 到 DOM 在 处 理 文 档 模 型 数据 时 非常 完善 和 强大 。 与 其 他 
XML 引擎 不 同 ，DOM 对 文档 节点 的 CURD 操作 是 彻底 及 完善 的 。 因 为 DOM 的 强大 及 出 
E, MIREA W3C 推荐 为 处 理 可 扩展 置 标语 言 的 标准 编程 接口 。 但 有 一 点 读者 需要 明 
日 的 是 ， 虽 然 PHP 可 以 使 用 DOM 来 处 理 文档 模型 ， 但 通常 来 讲 在 PHP 中 只 和 需要 读 取 即 可 
(数据 通常 保存 到 数据 库 )， 文 档 模 型 的 CURD 操作 一 般 处 理 界 面 UI 时 才 需 要 使 用 ， 例 如 
Jquery, ExtJs 等 框 染 都 提供 了 完善 的 DOM 文 持 。 

正 因 如 此 ，PHP 还 内 置 了 一 种 更 加 人 简单 和 多 用 的 XML 但 询 解 释 扩展 SimpleXML。 访 扩 
展 处 理 机 制 上 与 SAX 相似 ， 并 且 拥 有 SAX 快速 及 稳定 的 特性 ， 但 相对 SAX 来 说 ， 
SimpleXML 在 使 用 方式 上 更 加 简单 及 融 效 ， 所 以 SimpleXML 是 PHP 官方 推荐 使 用 的 XML 
解释 引擎 。 

3. SimpleXML 

SimpleXML 是 PHP Un 24 XML 解释 右 ， 使 用 SimpleXML 来 解释 XML KO 
效 和 稳定 的 。 如 果 只 需要 单纯 的 查询 XML 文档 ， 那 么 使 用 SimpleXML 是 一 个 非常 好 的 方 
Ze, SimpleXML 不 需要 像 SAX 引擎 那 样 使 用 多 个 方法 进行 捕捉 XML 信息 ， 也 不 需要 像 
DOM 那样 多 次 调用 相应 的 方法 遍历 文档 模型 。SimpleXML 将 XML 文件 映射 为 一 个 PHP 类 
对 销 ， 调 用 类 中 的 成 员 束 等 于 获取 XML 文档 中 的 节点 信息 。 下 面 将 结合 示例 代码 详细 介绍 
SimpleXML 的 使 用 。 

继续 以 前 面 创建 的 news.xml 文件 为 例 ， 使 用 SimpleXML 来 解释 并 提取 文件 中 的 数据 。 
首先 创建 一 个 类 ， 并 命名 为 SimpleXMLNews.class.php。 该 类 用 于 实现 所 有 SimpleXML 查询 
操作 ， 如 以 下 代码 所 示 。 


<?php 


class SimpleXMLNews { 
protected $data; 
function _ construct ($file) { 
Şdata=file_get_contents ($file); 
$this->data=$data}; 
} 
public function readXxml (){ 
$xmlObj=simplexml_load_string ($this->data); 
sdatas=array ();}; 
S10 
foreach ($xmlObj->row as Skey=>$value)t 
SE 
// 将 对 象 数 据 转 换 成 数组 ， 便 于 使 用 页 面 中 循环 
şdatas[$i]["title"]=Şvalue->title; 
$datas[$i]["content"]=$value->content; 
$datas[$i]["date"]=$value->date; 


; 


return datas; 


} 


R 


如 上 述 代码 所 示 ， 使 用 SimpleXML 处 理 XML 文档 将 变 得 非常 简单 。 整 个 步骤 只 需 将 
XML 文档 数据 传 入 到 simplexml_load_string 函数 ， 衣 函数 接受 字符 串 数 据 。 类 似 的 图 数 还 有 
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simplexml_load_file, simplexml_import_dom, similar_text 等 ， 这 些 函 数 都 是 用 于 载 入 数据 
的 ， 只 是 载 入 的 方式 有 上 所 不 同 。 

但 不 管 哪 种 方式 载 入 数据 ， 只 要 成 功 载 入 数据 ，SimpleXML 引 | 苟 束 会 日 动 将 XML 文档 中 
的 节点 映射 为 类 成 员 属 性 ， 开 发 人 员 可 以 像 普 通 的 面 问 对象 开 发 一 样 直 接 实例 化 对 象 〈“ 例 如 
$xmlObj->title 相当 于 获取 tite TALA), SCH rm XML。 但 为 了 方便 前 台 调 用 ， 所 以 在 上 述 
代码 中 将 $xmlObj 对 象 转换 为 数组 。 使 用 时 直接 调用 readXml 方法 即 可 ， 如 以 下 代码 所 示 。 


<?php 
require_once ("SimpleXxXMLNews.class.php"); 


$sSimpleXMLNews=new SimpleXMLNews ("news.xml"); 
Şdatas=$SimpleXxXMLNews->readXml (); 
foreach ($datas as Şkey=>Ş$value){ 
$times=date ("Y-m-d H:i:s","{Ş$value['date']}"); 
EC SE 
} 


通过 前 面 的 介绍 ， 相 信 读 者 已 经 能 够 领会 到 3 种 主流 XML 引擎 的 要 点 。SAX 解释 速度 
快 ， 适 用 于 数据 量 少 的 XML 文档 (300KB 以 下 )， 基 于 事件 驱动 的 函数 式 触 发 能 够 准确 捕 
捉 到 XML 文档 中 的 每 个 元 素 、 节 点 、 属 性 等 ，DOM OI W3C 推荐 的 文档 模型 处 理 
引 苟 ， 得 益 于 在 内 存 中 人 处理 数据 ， 所 以 能 够 提供 强大 的 CURD 文档 模型 操作 功能 ， 同 时 很 
好 地 避免 了 因 即 时 解释 造成 的 超时 ， 所 以 DOM 能 够 应 用 于 大 型 数据 文档 (前 提 是 服务 器 内 
存 足够 大 ); SimpleXML 是 一 款 小 巧 的 PHP 扩展 ， 只 需要 传 入 数据 ， 所 有 XML 的 处 理 过 程 
将 由 SimpleXML 目 动 完成 ， 开 发 人 员 只 需要 实例 化 对 象 即 可 。 

综 上 所 述 ， 如 果 需 要 快速 开发 ， 可 以 使 用 SimpleXML; 如 果 对 性 能 要 求 比较 苛刻 ， 可 
以 考虑 SAX; 如 果 文 档 数 据 比 较 大 (1024KB 以 上 )， 则 需要 使 用 DOM。 在 实际 应 用 开发 
中 ， 读 者 可 以 根据 需要 进行 选择 。 

另外 ，XML Reader 解释 引擎 也 是 一 球 高 性 能 的 XML 处 理 引 擎 ， 该 引擎 以 数据 流 的 方 
式 对 XML 进行 操作 ， 能 够 对 整个 文档 节点 进行 顺序 全 部 读 取 SAX 只 能 处 理 部 分 节点 )， 
并 且 对 大 数据 进行 了 分 段 处 理 〈 将 一 部 分 数据 存放 在 缓存 区 中 ， 一 部 分 在 内 存 中 进行 解 
释 )， 所 以 XML Reader 是 非常 优秀 的 XML 引擎 。 由 于 篇 幅 所 限 ， 这 里 将 不 再 深入 介绍 ， 感 
兴趣 的 读者 可 以 参考 PHP 官方 手册 。 


9.1.2 ”返回 XML 


DOM 可 以 创建 并 生成 静态 的 XML 文件 ， 但 在 实际 应 用 开发 中 通 间 使 用 动态 生成 。 动 
态 生成 可 以 实时 地 从 数据 库 中 获取 数据 ， 并 以 标准 的 XML 格式 输出 ， 第 三 方 应 用 (如 PHP 
应 用 、 手 机 终端 应 用 ) 调用 动态 生成 的 XML 文件 能 够 实现 远程 与 本 地 的 高 效 互 动 。 
ThinkPHP 内 置 了 许多 生成 XML 数据 的 类 库 ， 第 用 的 有 Action 控制 器 类 中 的 ajaxReturn 方 
法 、display 方法 以 及 xml_encode 扩展 水 数 。 下 面 分 别 介 绍 。 

1. ajaxReturn 方法 

ajaxReturn 方法 是 Action 控制 堪 基 类 中 用 于 返回 数据 的 方法 ， 该 方法 可 以 返回 Json、 
EVAL 序列 化 数据 以 及 通用 的 XML 标记 数据 。ajaxReturn 通 负 结合 Ajax 异步 请 求 来 使 用 ， 
但 也 可 以 用 于 设计 开放 API〔 网 站 接口 )。 
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ajaxReturn Æ% XML 是 非常 简单 的 ， 开 发 人 员 不 需要 定义 文档 节点 ， 甚 至 不 需要 了 解 
XML 文档 结构 ， 因 为 ajaxReturn 使 用 关联 数组 生成 XML 文档 模型 。 也 就 是 说 普通 数组 中 的 
键 名 (key) 对 应 XML 文档 中 的 节点 名 称 ; E (value) 对 应 节点 文本 。 假 设 PHP 数组 代 
但 如 下 所 示 。 


Şarray=array ( 
Vusertname "=> "JRM", 


Tage=->26, 
"sex"=>" H", 


Wemel ee koe0 Som, 
"address"=>"] 东 省 厂 州 市 Uz 


) 7 
使 用 ajaxReturn 转换 后 ， 将 得 到 标准 的 XML 格式 数据 ， 如 以 下 代码 所 示 。 


<?xml version="1.0" encoding="utf-8"?> 
<think> 
<status></status> 
<info></info> 
<data> 
<uUsername> 李 开 涌 </username> 
<age>26</age> 
<sex> E</sex> 
<email>kf@86055.com</email> 
<address> 广 东 省 广州 市 </address> 
</data> 
</think> 


ajaxReturn 方法 返回 结果 受 DEFAULT_AJAX_RETURN 配置 项 有 影响， 默认 值 为 json， 可 
选 的 值 有 json, eval 及 xml。 当 然 也 可 以 指定 ajaxReturn 方法 第 4 个 参数 值 为 xml， 强 制 输 
出 xm 数据 。ajaxRetun 方法 共 支 持 4 个 参数 ， 表 现形 式 为 ajaxReturn($data,$info,$status， 
$type)。 改 变 参 数值 将 直接 影响 输出 结果 ， 如 表 9-1 所 示 。 


表 9-1 ajaxReturn 方法 参数 


ZS 数 说 D 默 认 值 
data 需要 转换 的 PHP 数组 数据 T OR) 
info 提示 信息 Saar 
status 状态 TTA) 
type 生成 类 型 〈 可 选 的 值 有 json, eval, xml) json CÙ) 


Model 类 中 的 select 或 find 方法 返回 的 数组 原本 就 是 天 联 数 组 数据 ， 所 以 ajaxReturn 能 
够 很 好 地 将 结果 转换 为 XML 文档 ， 如 以 下 代码 所 示 。 


<?php 
// 本 类 由 系统 目 动 生成 ， 仪 供 测 试用 途 
class IndexAction extends Action { 
public function index (){ 
Şarticle=M ("Article"); 
Ş$rows=$article->field("content",true)->select()}; 


$this->ajaxReturn ($rows,"","","xml"); 
J 


R 
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ajaxReturn 最 终 会 将 数据 表 字 段 转 换 为 XML 子 节点 名 称 ， 字 段 值 将 转换 为 节点 文本 ， 
如 图 9-1 所 示 。 


访 XML 充 件 并 未 包 侣 任何 关于 的 样式 信息 。 文 档 剧 号 示 是 下 。 


<add Ser cebas /add_user> 
zadd time» 1454016% dd time» 


eëd user Cerbae fad User> 
<add_times 1346401704 < /add me: 
«lock verdon» 

«iter» 


图 9-1 ajaxReturn 生成 XML 结果 


震 要 注意 的 是 ajaxReturn 生成 的 XML 标签 名 称 是 受 目 定义 模型 字段 映射 Cé map) 影 
吧 的 《可 参考 本 书 第 7 章 7.1 市 )。 

2. display 方法 

display 方法 是 控制 器 与 视图 引擎 进行 交互 的 入 口 ， 系 统 内 置 的 视图 引擎 默认 将 模板 洽 染 
为 HIML， 通 过 display 方法 第 3 个 参数 可 以 改变 泻 染 方式 。display TIRRI 3 个 参数 ， 
表现 形式 为 display($templateFile,$charset,$contentType)， 参 数值 的 设 定 如 表 9-2 PZR. 


表 9-2 dsplay 方法 参数 


ZS X 说 H 默 认 值 
templateFile 默认 文件 路 笃 ， 不 需要 后 级 名 E OE) 

charset 输出 的 文件 编码 utf-8〈 可 选 ) 
contentType 输出 的 数据 类 型 html 《可 选 ) 


对 于 XML 数据 而 言 ， 只 需要 更 改 contentType 参数 值 即 可， 但 该 参数 值 只 能 对 当前 的 输 
出 产生 人 作用， 如果 和 需要 应 用 到 整个 项 目 ， 开 发 人 员 还 可 以 通过 修改 TMPL_CONTENT_TYPE 
配置 项 值 为 xml， 使 得 项 目下 的 所 有 display 输出 变 为 XML。 

需要 说 明 的 是 ， 虽 然 通 过 修改 配置 项 contentType 可 实现 XML 输出 ， 但 模板 文件 还 必须 
使 用 HTML 来 描述 ， 如 以 下 代码 所 示 。 


<?php 


class IndexAction extends Action { 
public function index (){ 
$article=M("Article"); 
Şrows=$article->field("content",true)->select()}; 
sthis->assign ("list",S$rows); 
$this->display ("index_ xml", 'utf-8', 'text/xml'); 


j 


?> 


如 上 述 代码 所 示 ，index_xml 的 真实 文件 名 为 index_xml.html， 所 以 开发 人 员 需 要 在 tpl 
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/Index/ 目 录 下 创建 该 文件 ， 代 码 如 下 所 示 。 


<?xml version="1.0" encoding="utf-8"?> 


<root> 
<status></status> 
<info></info> 
<data> 
<volist name="list" id="vo" key="k"> 
<rOow LCSW <l=-= {6k }==> > 
<id><!-—-{$vo.id}--></id> 
<reicle>< l- $yo, Ciele} =—></r Lre le> 
<user><!-—-{Ş$vo.add_user}--></user> 
<time><!-—-{$vo.add_time}--></time> 
</row> 
</volist> 
</data> 
</root> 


可 以 看 到 ， 使 用 display 输出 XML 与 上 默认 的 HTML 并 没有 区 别 ， 模 板 引 擎 中 的 所 有 标 
签 、 语 法 、 变 量 等 都 可 以 使 用 。 这 残 给 XML 开 友 市 来 非常 灵活 的 特性 ， 例 如 可 以 方便 过 历 
数据 、 设 置 日 定义 标签 、 目 定义 属性 每 。 最 终 的 运行 效果 如 图 9-2 br, 


该 XML 文件 并 未 包含 任何 关联 的 样式 信息 。 文 档 树 显示 如 下 。 


<roaot> 
<status => 
<=info/> 
- data: 
— <row id="1"> 
<id>1</id> 
<title> 库 克 就 革 果 地 图 缺陷 向 用 户 章 菊 : 未 提供 一 流体 验 </title> 
suser» Celba= /user> 
=<times1346401694 =< /times 
/row> 
-<row id="2"> 
<jds2</id> 
<title> 索 尼 投 资 6.44 亿 美元 成 为 奥 林 巴 斯 第 一 大 股东 </title> 
<UsSer> ceba </user> 
<time> 1346401704 = /time> 
/ToOW> 


图 9-2 display 输出 xml 


需要 注意 的 是 ， 由 于 XML 对 标记 的 处 理 比 HTML 要 严格 ， 所 以 在 使 用 display 输出 
XML 时 ， 不 能 有 其 他 的 非 XML 数据 被 输出 《例如 在 动作 或 模板 中 输出 调试 信息 、 性 能 信息 
或 者 开局 LAYOUT_ON 布局 选项 等 )， 否 则 由 于 标记 谥 出， 导致 浏览 髓 解释 异 香 。 

3. xml encode 函数 

xml_encode 函数 用 于 快速 生成 XML 数据 ， 功 能 上 类 似 于 ajaxReturn， 这 两 者 所 生成 的 
XML 标签 名 称 和 标签 文本 都 是 根据 关联 数据 键 名 与 键 值 而 来 的 。 不 同 之 处 在 于 xml_encode 
可 以 在 MVC 所 有 功能 类 库 中 进行 使 用 ， 包 括 目 定义 扩展 、 目 定义 函数 、 目 定义 模型 等 。 而 
ajaxReturn 只 能 在 控制 右 动 作 中 进行 使 用 。 

另外 ，xml_encode 虽然 不 能 自 定 XML 文档 标签 ， 但 允许 自 定 义 根 节 点 。xml_encode DÉI 
数 支 持 3 个 参数 ， 表 现形 式 为 xml_encode($data, $encoding, $root)， 参 数值 如 表 9-3 所 示 。 
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表 9-3 xml encode 函数 参数 


ZS Sr 说 D 默 认 值 
Data 用 于 生成 XML 数据 的 关联 数组 TE 

encoding, 输出 的 文件 编码 utf-8〈 可 选 ) 
root XML 文档 模型 的 根 节 点 名 称 think〈 可 选 ) 


xml_encode 函数 的 使 用 非 音 简单 ， 和 普通 的 目 定 义 函 数 没 什么 区 别 ， 如 以 下 代码 所 示 。 
<?php 
class IndexAction extends Action { 
public function index (){ 
Şarticle=M ("Article"); 
Ş$rows=$article->field("content",true)->select()}; 
header ('Content-Type:text/xml; charset=utf-8'); 
exit (xml_encode ($rows, 'utf-8','root')); 


} 


事实 上 ，ajaxReturn 方法 本 质 上 也 是 调用 xml_encode 轴 数 的 ， 所 以 在 实际 应 用 开发 中 如 
果 只 需 单一 地 生成 xm 数据 ， 完 全 可 以 使 用 xml_encode 1# ajaxReturn 方法 。 

通过 前 面 内容 的 介绍 ， 可 以 看 到 使 用 display 方法 处 理 xml 是 最 灵活 和 强大 的 ， 如 果 配 
合 后 面 草 布 将 介绍 的 静态 绥 存 功能 ， 可 以 将 结果 进行 缓存 ， 以 提高 数据 啊 应 速度 。 

display 是 控制 器 基 类 show 的 数据 交互 方法 ，show 方法 直接 调用 模板 引擎 中 的 view 
ERE action 类 中 ) 接口 ， 根 据 面 问 对 和 象 原 则 ， 开 发 人 员 可 以 直接 调用 show 方法 进行 处 理 
数据 〈 需 要 使 用 连贯 操作 ， 即 $this->view->show ())。 

通过 这 前 面 内 容 的 学 习 ， 相 应 读者 能 够 使 用 PHP 轻松 地 生成 XML 文件 ， 以 及 使 用 内 置 
的 XML 引擎 解释 XML 数据 了 。XML 只 是 一 种 数据 规范 ， 接 下 来 将 学 习 另 外 一 种 非常 流行 
的 数据 规范 。 


9.1.3 REl Json 


Json 是 一 种 轻 量 级 的 数据 交换 格式 ， 它 的 英文 全 称 为 JavaScript Object Notations Json 使 
用 JavaScript 对 象 语法 ， 能 够 以 价 单 的 语法 格式 对 JavaScript 对 象 进行 表 示 。 虽 然 其 中 市 有 
JavaScript 名 称 ， 但 事实 上 Json 是 一 种 独立 的 开放 式 的 数据 交互 格式 ， 在 主流 的 语言 或 框架 
中 均 提 供 有 相应 的 Json 序列 化 及 反 序 列 化 类 库 。 

Json 与 XML 类 似 ， 都 是 进行 数据 交互 的 利用 格式 ， 并 且 都 文 持 本 地 与 Ajax 远程 通信 
Json 需 借助 于 Jsonp 协议 )， 但 是 Json 不 需要 定义 标签 ， 数 据 得 到 极 大 地 压 因 ， 所 以 相对 来 
讲 Json 比 XML 传输 效率 更 加 出 色 。 同 时 由 于 Json 本 质 上 是 一 种 序列 化 文本 ， 所 以 Json 不 
需要 特定 的 解释 引擎 ， 在 JavaScript 脚本 中 其 至 只 需要 使 用 for 语句 或 evalOR ZS ol LA är 
据 进行 提取 。 

下 是 由 于 Json 的 简单、 轻巧 及 稳定 ， 近 年 来 已 被 各 大 网 站 纷纷 使 用 ， 典 型 的 应 用 有 网 
站 开放 API、UI 前 后 台 分 离 、Ajax 异步 请 求 等 。 对 于 PH 开发 人 员 而 言 ， 使 用 Json 是 非常 
方便 的 ，PHP 5.x A DU S json_encode 及 json_decode 函数 ， 用 于 实现 数组 数据 友 列 化 
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及 反 序 列 化 。 下 和 面 将 结合 ThinkPHP 深入 介绍 Json 在 MVC 开发 中 的 实际 应 用 。 

1. json_encode 函数 

json_encode K% PHP 5.x 内 置 的 一 个 标准 函数 ， 该 函数 用 于 实现 对 数组 进行 Json 格 
式 序 列 化 。json_encode 的 使 用 非常 简单 ， 如 以 下 代码 所 示 。 


<?php 


class IndexAction extends Action { 
public function index (){ 
Şarray=array ( 
"name"=>" FJR", 
"age"=>20; 
"email"=>"kfQ86055.com" 
); 
header ('Content-Type:text/html; charset=utf-8'); 
exit (json_encode ($array)); 
} 

} 

json_encode 只 能 接受 utf-8 编码 的 数据 ， 如 果 当 前 文档 不 为 utt-8， 需 要 使 用 iconv SR 
数 进行 编码 转换 。 上 述 代码 的 转换 结果 如 下 代码 所 示 。 

{"name":"\u674e\u5f00\u6e67", "age":20, "email":"kfQ@86055.com"} 

可 以 看 到 ，json_conde 函数 将 标准 的 数组 信息 转换 为 了 紧凑 的 字符 串 信 息 ， 这 种 被 
序列 化 后 的 字符 串 也 是 使 用 键 值 对 来 描述 的 ， 例 如 age 的 键 值 为 20， 使 用 “:” 进 行 分 
隔 〈 相 等 于 数组 中 的 =>)， 多 个 键 值 对 之 间 使 用 “,” 分 开 。 可 以 看 到 Json 没有 标签 或 节 
点 的 概念 ， 所 有 数据 都 是 序列 化 后 的 键 值 对 ， 这 无 论 对 开发 效率 还 是 传输 效率 都 是 非常 
A o 

2. ajaxReturn 方法 

ajaxReturn 方法 前 面 已 经 有 过 深入 地 介绍 ， 在 此 不 再 重 述 。ajaxReturn 默认 情况 下 束 是 
返回 Json 格式 数据 ， 能 够 方便 地 与 前 台 Ajax 实现 高 效 互 动 ， 如 以 下 代码 所 示 。 


<?php 


class IndexAction extends Action { 
public function index (){ 
Şarray=array ( 
"name "=>" EFR", 
"age"=>20, 
"email"=>"kf086055.com" 
) 7 
$this->ajaxReturn ($array) ; 
} 
i 
?> 
上 述 代码 的 运行 结 末 如 下 代码 所 示 。 
人 ESaEUSLIOTIEOL DGSE SS TISSUENOSENNNU6GEGYT vase20, Wenewu ll We wd Gon h) 


与 生成 xml 数据 一 样 ，status、info 及 data 这 3 个 Json 序列 键 都 是 系统 自动 生成 的 ， 方 
便 Ajax 编程 。 其 中 status 表示 状态 ; info 表示 提示 信息 ; data 用 于 存放 序列 化 后 的 数组 数 
Ho Json 与 数组 一 样 ， 支 持 多 维 网 套 ， 例 如 data 键 对 应 的 值 就 是 一 个 Json 序列 ， 所 以 直接 
输出 select 结果 也 是 允许 的 ， 如 以 下 代码 所 示 。 


<?php 
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class IndexAction extends Action { 
public function index (){ 
sarticle=M ("Article"); 
Ş$rows=$article->field("content",true)->select(); 


$this->ajaxReturn ($rows); 


} 


3. 实现 Jsonp 

前 面 已 经 提 到 过 Json 实现 本 地 和 远程 Ajax 通信 是 基于 Jsonp 的 。Jsonp 是 一 种 通信 方 
式 ， 这 一 点 初学 者 往往 将 其 与 Json 混淆 ， 事 实 上 它们 是 两 个 完全 不 同 的 概念 ， 简 单 地 说 
Jsonp 是 通信 方式 ，Json 是 数据 格式 。 用 过 Ajax 编程 的 读者 应 该 了 解 Ajax 默认 情况 下 是 不 
允许 跨 域 请 求 的 ， 所 以 要 实现 与 远程 〈( 跨 域名 ) 的 xm, JavaScripts htm 等 数据 进行 通信 ， 
必须 设置 请 求 代 理 ， 例 如 实现 ifiame、 本 地 储存 等 或 者 添加 同 源 组 条 略 〈 适 用 于 
Windows)。 好 在 现在 的 JavaScript 耐 问 对 和 象 编程 全 部 封装 了 这 些 操 作 〈( 包 括 各 种 框架 ， 例 如 
jquery、extjs 等 )， 开 发 人 员 和 需要 做 的 步 骂 其 实 非常 少 。 但 Json 不 同 ，Json 是 一 种 序列 化 的 
数据 (相当 于 纯 文 本 )， 所 以 本 号 不 具备 跨 域 通信 的 能 力 ( 没 有 用 于 代理 回调 的 标记 )， 为 了 
解决 Json 路 域 通 信 的 问题 ， 所 以 Jsonp 束 应 运 而 生 了 。 

Jsonp 请 求 数据 时 ， 会 在 url 8 cakebale 回调 函数 ，Json 数据 必须 回应 该 函数 ， 才 能 统 
过 浏览 器 的 安全 策略 ， 实 现 跨 域 通 信 。 所 以 要 让 json_encode 序列 化 的 Json 能 够 用 于 跨 域 处 
理 ， 只 需要 加 入 回调 函数 标识 即 可 ， 如 以 下 代码 所 示 。 


<?php 
function jsonp_encode ($data, $info, sstatus)t 
Şresult ["data"]=Ş$data; 


Se le ne 

sresult["status"]=$status; 

if (empty(s$_GET["callback"™])) { 
$json=json encode ($result); 

}else { 
//Jsonp 跨 域 通信 
$json=$_GET["callback"]."(".json encode ($result)."™)"; 


} 
header ("Content-Type:text/html; charset=utf-8"); 


exit ($json),; 


} 


EK 


将 上 述 目 定义 函数 保存 到 Common.php 文件 中 ， 在 控制 右 动 作 中 直接 调用 即 可 ， 如 以 下 
代码 所 示 。 


public function index (){ 


Şarray=array ( 
"name "=>" 李 开 Y It: 
"age"=>20, 
"email"=>"kf@86055.com" 

); 

jsonp_encode ($array); 
} 
在 Ajax 异步 请 求 时 上 只 需要 为 callback 传 入 回调 标识 名 称 ， 即 可 实现 Jsonp 通信 。 假 设 传 


HH callback 标识 名 称 为 datas， 那 么 最 终 的 Json 数据 格式 如 下 代 公 所 示 。 
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datas ({"name":"\u674e\u5f00\u6e67","age":20,"email":"kf@86055.com"}) 
Jsonp 能 够 轻松 地 实现 Ajax 异步 请 求 ， 关 于 Jsonp 的 实际 应 用 ， 接 来 下 将 结合 JavaScript 
进行 简单 介绍 。 


9.1.4 使 用 Json 及 Jsonp 


Json 能 够 应 用 在 许多 场合 ， 包 括 PHP 本 喘 耽 提供 了 对 Json 的 反 序 列 化 操作 。 当 然 ， 
Json 更 多 地 应 用 在 Ajax 或 者 各 种 终端 与 网 站 的 通信 上 《例如 手机 程序 、 果 面 客户 端 、Flash 
等 )。 接 下 来 首先 介绍 PHP 内 置 的 json_decode 函数 ， 然 后 再 结合 JavaScript 讲解 Json 在 网 站 
UI 设计 上 的 实际 应 用 。 

1. json_decode 函数 

json_decode 函数 是 PHP 内 置 的 标准 函数 ， 访 函数 用 于 反 序列 化 Json 数据 。json_decode GI 
数 共 文 持 3 个 参数 ， 表 现形 式 为 json_decode ($json, $assoc, $depth)， 参 数值 如 表 9-4 PTR. 


表 9-4 json decode 函数 参数 


5 数 说 明 默 认 值 
Json Json 序列 化 数据 TOVE) 
Assco 当 assco 为 true HHRH, AUREX false〈 可 选 ) 
Depth 反 序 深度 null 〈 可 选 ) 
json_decode 的 使 用 非常 徐 单 ， 如 以 下 代码 所 示 。 
<?php 


class IndexAction extends Action { 
public function index (){ 
$str='{"name":"\u674e\u5f00\u6e67","age":20,"email":"kf@86055.com"}!'; 
header ('Content-Type:text/html; charset=utf-8'); 
Şarray=json_decode ($str,true); 
dump (Sarray); 
k 


2 


上 述 Json 数据 序列 化 的 结果 如 下 代码 所 示 。 


array (3) { 


[name] > stringo) “= Ha" 
[Vage] => imr (20) 
["email"] => string(12) "kfQ86055.com" 


} 

json_decode 虽然 大 多 数 情况 下 都 可 以 实现 对 Json 反 序 列 ， 但 需要 注意 的 是 Json 数据 编 
但 必须 为 utf-8。 男 外 ，json_decode 在 反 序列 少量 数据 时 ， 通 党 情况 下 是 可 菲 的 ， 但 在 反 序 
列 大 量 数 据 ， 尤 其 包含 大 量 中 文 或 特殊 字符 时 ，json_decode SEME. PRE, Json 在 传 
送 大 量 数 据 时 本 里 是 存在 淤 出 的 ， 所 以 在 设计 时 服务 问 要 尽量 避免 输出 大 量 Json 数据 。 如 
果 有 此 需求， 建议 使 用 XML 或 者 后 和 面 章 市 将 介绍 的 wsdl。 

2. Ajax 异步 请 求 

Ajax 是 近年 来 发 展 迅速 的 一 种 网 页 UI 编程 技术 ， 几 乎 所 有 SNS 类 型 的 网 站 都 离 不 开 
Ajax， 当 然 普 通关 型 的 网 站 使 用 Ajax 也 能 明显 提高 用 户 体验 。PHP 开发 人 员 虽 然 大 多 数 情 
况 下 只 负责 后 台 的 程序 设计 ， 但 如 采 不 了 解 Ajax 束 很 难 与 前 端 设计 进行 民 好 的 沟通 。 前 闹 
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与 后 人 台 本 来 瓯 相互 依赖 的 ， 前 端 可 以 不 了 解 后 台 设 计 ， 但 需要 掌握 怎样 处 理 后 全 输出 的 结 
果 ; 同样 ， 后 从 开发 可 以 不 必 精 通 前 端 设 计 ， 但 需要 理解 怎样 才能 输出 符合 前 端 设 计 要 求 的 
数据 格式 。 这 就 是 典型 的 前 后 台 分 离 设 计 ， 如 果 彼 此 配合 恰当 ， 对 整个 项 目的 开发 无 论 是 代 
但 质量 还 是 开 友 效率 都 是 质 的 提升 。 

要 达到 前 后 台 分 离 设计 ， 使 用 Json 通信 最 好 不 过 了 ， 因 为 主流 的 UI 设计 框架 都 对 Json 
提供 了 完善 的 文 持 ， 下 面 将 以 Jquery 为 基础 ， 详 细 介绍 Ajax 与 Json 的 开发 过 程 。 

(1) Jquery 

Jquery 是 当前 最 主流 的 UI RIER, bai (pg elt Ar, AE ExtJs， 但 由 于 其 轻 
巧 、 敏 捷 及 兼容 性 全 面 ， 近 年 来 纷纷 被 国内 外 大 型 网 站 所 采用 。Jquery 是 高 效 的 ， 这 种 局 效 
不 仅 是 运行 效率 ， 还 有 开发 效率 ， 这 氮 是 毋庸 置疑 的 。 假 设 有 一 个 Ajax mR, ARIU 
ees 


<script type="text/javascript"> 


function getResult () { 
var url = "ajaxServlet?action=send"; 
if (window.XMLHttpRequest){ 
req = new XMLHttpRequest () ; 
yelse if (window.ActiveXObject) 
{ 
req = new ActiveXObject ("Microsoft .XMLHTTP"); 
l 
if (req) 
l 
TeeGee open ("On E /tesr mml"; Crug) 7 
req.onreadystatechange = complete; 
req.send (null); 
l 
| 
/* 分 析 返 回 的 XML 文档 */ 
function complete (){ 
if (req.readyState == 4) 
{ 
if (req.status == 200){ 
var type = req.responseXML.getElementsByTagName ("type_name"); 
var str=new Array (); 
for(var i=0;i<type.length;i++){ 
Seel Ell EE ee Ee Ee E: 
document.all[td].innerHTML+=str[i]+"<BR>"; 
} 
} 
} 
i 
</script> 


上 述 代码 用 于 异步 请 求 xml 数据 ， 如 来 使 用 jquery KAA, MAREEA FR 


<script type="text/javascript"> 


Ş$.ajax({ 
urls "aJjax/ testo xnl"; 
GEaaS s Vmi", 


e Eeer 
success: function (xml) { 
$ ("AUTHOR", xml) .each (function (id) { 
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/ / AUTHOR = $ ("AUTHOR", ¥ml) o GET (ia) 5 
alert ($ (thais) -cniladren ("FIRSTNAME") text ()) s 
alert ($ (thais) children (“LASTNAME*) ,Text ()); 


} 
kk: 
</SCript> 


可 以 看 到 ， 使 用 Jquery 后 ， 无 论 是 代码 质量 还 是 数量 ， 痢 发 生 了 极 大 变化 ， 开 发 人 员 只 
需要 填充 $.ajax 方法 属性 即 可 。 如 果 处 理 Json 数据 ， 代 码 将 变 得 更 加 简单 ， 因 为 Jquery 从 设 
计 之 初 束 非常 重视 轻 量 级 的 Json 数据 ， 提 供 了 快捷 选择 占 $.getJSON。 

Jquery 易学 多 用 ， 通 过 上 述 代 码 演 示 ， 谈 者 应 该 对 $ 人 符号 非 第 加 悉 ， 这 种 人 符 亏 称 为 
选择 符 ， 后 面 的 标识 称 为 方法 《或 称 动 作 )。 这 里 所 说 的 选择 符 与 css 设计 中 的 选择 和 从 
原理 是 一 样 的 ， 都 是 用 于 选择 指定 的 HTML 文档 模型 中 的 节点 ， 这 个 过 程 称 为 DOM 
选择 操作 。 

总 之 ，Jquery 是 强大 的 ， 配 合 数 以 万 计 的 扩展 插件 ，Jquery 在 前 问 设 计 中 几乎 无 所 不 能 。 
由 于 本 书 并 非 介 绍 UI 设计 的 ， 旋 者 如 需要 更 深 一 步 学 习 Jquery， 和 需要 日 行 合 阅 相关 图 书 或 资 
料 ， 这 里 为 了 方便 演示 Json 数据 处 理 过 程 ， 只 需要 理解 Jquery 的 异步 请 求 功能 即 可 。 

(2) 寞 步 请 求 Json 

异步 请 求 即 通常 所 说 的 Ajax (Asynchronous JavaScript and XML), FH ÆN T Aim- ri 
台 的 XML 异步 通信 而 设计 的 ， 但 现在 的 Ajax 早已 与 XML 没有 特定 关系 ， 它 不 仅 用 于 异步 
处 理 XML, LHT Json, Eva 等 所 有 序列 化 数据 ， 并 且 性 能 上 比 XML 更 为 出 
Eo Jquery 对 Ajax 的 封装 是 彻 抵 及 灵活 的 ， 开 发 人 员 甚 至 不 需要 了 解 XMLHttpRequest 任何 
知识 ， 也 能 够 开发 出 稳定 强大 的 Ajax 应 用 。 下 和 耐 将 通过 一 个 示例 演示 Jquery 中 的 Json 处 理 

Ajax 开 有 大体 上 共 分 为 两 个 步 又 ， 即 后 台 及 前 靖 。 首 先 在 ThinkPHP 中 设计 后 侣 ， 这 里 
为 了 方便 读 示 ， 将 创建 一 个 显示 新 闻 标 题 的 页 面 ， 代 人 码 如 下 所 示 。 

S 文章 列表 

* Enter description here ... 
E 

public function article()1 


Sobi-M(fArticleii: 
Şrows=$0b]=>0rder ("id Cesc") =>select () ; 


ŞthLs=>assign (V LLsSt",; $TrOwWS) 5 
ŞthaLrs=>cisplay (|); 


j 
article 动作 用 于 获取 文 草 列表 数据 ， 并 将 数据 输出 到 视图 模板 中 ， 相 应 的 article.html 视 
图 模板 文件 代码 如 下 所 示 。 


<style> 
/* 列 表 样 式 */ 
</style> 
<div class="main"> 
et 
<ul elass=" article"> 
<volist names" list" 10="yo"> 
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lee EE E 
<a href="javascript:void();" onClick="detailArticle(<!--{$vo.id}-->);"><!-- 
{$vo.title}--></a> 
<span> [<a href="javascript:void();" onClick="deleArticle (<!--{$vo.id}-->);"> 


删除 </a>】</span> 


<div class="content" style="display:none"> 


</div> 
Lf LLS 
</volist> 
</ul> 
el SIN 


<ul elags= tpost Ee 


<form name="form1" onSubmit="return false;"> 
人 


<LI> 
<label> 
<textarea name="content" 
</label> 
<label> 


id="content™" 


cols="45" rows="5"></textarea> 


<input type="submit" onClick="inputArticle();" name="button" id="button" value=" RR "> 


</label> 
/> 
</form> 
SE 
</div> 


最 终 的 模板 视图 效果 如 图 9-3 PR QE 
万 的 内 容 设 计 好 布局 文件 )。 


r 
J 


需 开 启 LAYOUT_ON 布局 选项 ， 并 按照 6.4.3 


"EE 
DEE CC Ia WS adi 
fflsdfsd 【删除 ] | 
Wm (Et 网 站 公告 
Wwwwww RER] BARNAT ， 欢 这 率 访问 预 的 
读 写 分 离 测试 【删除 a, een egen 
读 写 分 离 测 试 [删除 ] on 
读 写 分 离 测试 【删除 ] SE | 
内 容 存 文本 , 啊 啊 (HR ONEN 
内 容 存 文本 【删除 ] 
人 dfsdf 【删除 ] 
李 磨 兰 称 仍 未 找到 适 音 称 动 互 联网 的 商业 模式 【删除 ] 


HATER F acebook pM RE MER] 
TERUR REGISSEN ` 素 居 ITV 对 价格 千 满 【删除 ] 
夏普 称 会 制造 足够 iPhone5 所 需 的 显示 屏 【删除 ] 

素 尼 投资 6.44 亿 美元 成 为 奥 林 巴 上 类 第 一 大 股东 (Së) 

库 克 就 单果 地 图 缺陷 向 用 户 道 航 : 未 提 殿 一 访 休 验 【删除 ] 


CD 


图 9-3 article.html 模板 视图 效果 
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设计 好 视图 之 后 ， 接 下 来 就 需要 使 用 Json 异步 处 理 数据 了 ， 本 例 中 需要 的 效果 共 分 

为 3 大 部 分 : 首先 是 用 刀 单 击 新 闻 标 题 ， 将 会 在 新 闻 标 题 下 方 显 示 新 闻 的 详细 内 容 ; 当 
用 户 单 击 【删除 】 链 接 ， 将 删除 当前 单条 新 闻 ;， 当 用 户 在 标题 及 文本 框 中 输入 完 内 容 并 
提交 时 ， 策 要 使 用 卉 步 处 理 提交 到 后 人 台 ， 最 后 无 刷新 返回 到 当前 页 面 。 下 面 首先 实现 显 
不 新 闻 评 细 数 据 。 

/** 

* 单 篇 文章 详情 

s macer description NSLS ooo 

E 

public function detailArticle(){ 

Şobj=M ("Article"); 

Scara l Viet ]=$_CET]| UL60W] y 

Şrows=$0obj->where ($data)->find(); 


LE (Sows [vicv|) 1 
jsonp_encode ($rows, "数据 获取 成 功 "，1) ; 
}else{ 


jsonp_encode (null, "数据 获取 失败 "，0) ; 
} 


} 
"230 7 mem DI TI NC ée ber DT, AA detailArticle MWAK Z. ARIA kr, 


<script language="javascript" src="http://t.beauty-soft.net/js/jquery.js"></script> 
EEGEN 
/xx 点 击 显示 详 细 新 闻 */ 
function detailArticle (1Idq) { 
Saga (i 
// 跨 域 需 要 使 用 Jsonp 
url: 'http://127.0.0.1/tp3_demo/index.php/Index/detailArticle?id='+id+"&callback=?", 
ToS GE 
dataType: "json", 
cache: false, 
success: function(data) { 
if (data.status) { 
$ ("#1i_"+id+" .content") .html (data.data.content); 
S (Weli Wich eontentni.toecleii: 
}else{ 
alert (data.info); 


recura e 


LI: 


</script> 


如 上 述 代码 所 示 ， 在 使 用 Jquery 前 必须 首先 引入 框 染 核心 类 库 。 在 detailArticle 函数 中 
使 用 $.ajax 选择 器 获取 远程 ( 跨 域 ) 数据 。 这 里 为 了 方便 介绍 ， 使 用 127.0.0.1 代替 远程 地 址 
(本 地 域名 为 tp.localhost)， 在 实际 应 用 开 友 中 只 需要 蕉 换 为 真实 的 域名 即 可 。 最 终 的 运行 效 
果 如 图 9-4 所 示 。 

由 于 使 用 Jsonp 进行 异步 获取 数据 ， 所 以 Ajax 异步 通信 变 得 简单 。 与 种 见 的 同 域 Json 
通信 不 同 ，Jsonp 通信 时 会 在 callback 参数 后 附加 上 随机 函数 ， 通 过 Firebug Gë, OR 9-5 
PIR o 
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欢迎 光 | 临 我 的 博客 


李彦宏 称 仍 未 操 到 适合 穆 动 豆 联 网 的 商业 模式 (bi 


网 易 科技 讯 9 月 3 日 消息 ， 据 国外 搬 体 报道 ,百度 创始 人 ， 重 事 发 芷 首席 执行 让 李彦宏 有 昨 ass 
天 在 所 坦 福 大 学 举行 的 中 国 20 太 会 (China 20 conlerence ) 长 表 主 是 演讲 时 称 ， 百 度 志 一 
他 仍 未 技 吕 从 称 动 瑟 联 网 赚钱 的 最 好 方式 , 但 他 玉 示 将 副 心 汰 试 各 种 可 能 的 商业 槛 式 。 李 ”网站 公告 

彰 宏 证 通电， 虽然 夸 动 百联 网 带 来 的 党 收 占 百 度 沉 收 总 总 的 由 村 直到 10% ， 介 百度 已 将 的 种 的 博 放 开通 了 ,家 迎 率 访问 珀 的 博 | RRE 

瑟 台 的 研发 既 费 投 和 到 该 领域 ， 他 还 指出 , 自 百 讼 流年 推出 移动 产品 后 . 来 自 移动 写 联 网 。 这 更 对 总 嫩 我 的 学 习 成 晶 ,不 定时 地 沽 表 一 些 

的 营 收 增长 了 至 小 三 倍 。 李 次 家 在 演讲 时 称 : "说 训话 Eanpsgrëisinteréieg "ën, Eat, SR 

的 主要 萌 收 来 源 ,我 自己 也 二 清楚 ， 我 也 趟 意 看 去 弄 酒 吾 ,因为 我 们 如 洲 黎 动 广告 业 各 还 Ss, 

有 很 大 的 进一步 壳 善 的 空间 .“ 百度 盖 有 中 国 最 习惯 使 用 的 楼 索引 散 ， 志 月 早 些 时 候 推 出 
了 面向 Android 设 备 的 移动 浏览 图 ， EREM A, 互联 网 的 门户 。 提 市场 研 
究 公 司 iResearch 的 数据 ,去 年 百度 移动 广告 在 中 国 的 营 收 达到 24 亿 元 人 民 币 ,同比 增长 。。 MAAMA 
了 一 倍 以 上 。 百 庶 今 年 7 月 曾 表示 , 它 桂 为 新 浪 的 移动 门户 网 站 提供 所 未 服务 ， 此 前 百 廊 

同 迹 在 其 搜索 结果 中 蝇 示 与 关 岳 词 相 关 的 新 浪 微 博 内 容 ， 百 度 还 推出 了 其 它 有 一 利 澡 力 的 

ES, RS, TH, FEAET. 品 然 他 看 好 牧 动 互联 网 的 未 来 ， 但 目前 该 顿 域 

的 机 入 傅 远 太 于 产 出 ， 恺 说 : 对 我 们 来 说 ,找到 黎 动 互联 网 的 合适 商 业 模式 非常 重要 ， 

我 们 如 道 用 户 执 避 移 动 百联 网 ,我们 必 洒 潢 足 他 们 的 需求 ，" 在 美国 纽 交 所 昨天 交易 中 ， 

GO? 10 por, . et-28H Lia, PELLER SE SIT 

3. (HF) 


机 构 调 查 称 Facebeck 中 国 用 户 圳 超 6000 万 [删除 ] 

苹果 流 嵌 体 电 台 服 务 通 障碍 ` 标 尼 全 TV 对 价格 千 短 (Zb: 
夏普 称 会 制造 足 通 iPhone5 所 需 的 显示 屏 (Sr 

素 尼 投资 8.44 亿 美元 成 为 奥 林 巴 断 第 一 大 妥 东 。[ 删 陈 】 

库 克 就 苹果 地 图 缺陷 向 用 户 道 艇 : 未 提供 一 流体 验 【期 除 ] 


图 9-4 ”远程 获取 文章 数据 


-= 1351399661340 
callback jsonp1351398542068 


id ¢ 


pa 


图 9-5 Jsonp 跨 域 异步 通信 


接 下 来 将 实现 删除 文章 功能 。 用 户 单 击 【删除 】 链 接 ， 将 触发 deleArticle 脚本 函数 ， 代 


但 如 下 所 示 。 
/* 氮 击 删除 新 闻 */ 
function deleArticle (Idq) { 
if (confirm(" 真 的 要 删除 吗 ? ")){ 
// 当 前 域 下 不 需要 回调 函数 
var url="/index.php/Index/delArticle"; 
$.getJSON (url, {'id':id},function (data) { 
if (data.status) { 
// 后 从 删除 成 功 ， 移 除 界 面相 应 的 新 闻 
$ ("#1i n+id) .remove () ; 


}else{ 


alert (data.info); 


recura rcalgez 


Di 
}else{ 
EE Ee 


} 

这 里 使 用 了 getJSON 方法 ， 该 方法 是 Ajax 的 子 方法 ， 只 能 处 理 单纯 的 Json 数据 格式 。 
删除 数据 是 一 项 特殊 的 操作 ， 在 实际 应 用 开发 中 还 需要 结合 系统 认证 权限 来 处 理 数据 的 删 
除 ， 这 里 为 了 方便 演示 ， 和 省 略 了 相关 操作 。 
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接 下 来 将 继续 使 用 Json 实现 数据 的 提交 。 当 用 户 在 页 面 下 方 的 文本 框 中 输入 标题 及 内 
容 ， 单 击 【 提 交 】 按 钮 将 触发 inputArticle 脚本 函数 ， 该 函数 代码 如 下 所 示 。 


/ /发表 文 章 
function inputArticle(){ 
二 


alert (" 请 填写 数据 ") ; 
return false; 
} 
// 当 前 域 不 需要 回调 函数 
var url="http://tp.localhost/index.php/Index/postArticle"; 


alert (wl) p 
$.post (url, {'title':$("#title").val(), 'content':$("#content").val()}, 
function (data) { 
if (data.status){ 
// 数 据 发 送 成 功 ， 在 界面 上 无 刷新 添加 文章 标题 
S (W main EE EE EE 
onclick="'detailArticle("+data.data.iqd+");" 
有 
onclick-'deleñrticle("tdata.data.idi"),;' href- javascript:void(); -Mei </a- 1 </span <div 
style='display:none' class='content'></div> </li>"); 
}else{ 
alere (cers. LALO) S 
Í 
Le 'json'); 


} 

inputArticle AKIL J EERE, ERRER EHS. post 方法 提交 数据 ， 并 且 强 制 声 
明 回 调处 理 数据 为 Json 格式 (默认 txt 格式 )。 

最 终 的 体验 效果 是 ， 用 户 提 交 数 据 后 将 感觉 不 到 曾经 离开 过 当前 页 面 ， 但 界面 上 的 文章 
数量 却 发 生 了 改变 ， 这 正 是 Ajax 的 核心 理念 ， 也 是 Web 2.0/3.0 最 显著 的 特性 之 一 。 

上 述 例子 完美 地 演示 了 Json 与 Jsonp 的 实际 应 用 ， 读 者 可 以 在 此 基础 上 继续 完善 。 例 如 
加 入 权限 判断 ， 返 回 的 文章 数据 增加 分 类 、 友 表 时 间 、 友 表 用 户 等 ， 以 提 融 对 Json 及 Jsonp 
的 认识 。 


9.2 ThinkPHP 函数 库 


ThinkPHP 内 置 了 许多 系统 图 数 库 ， 使 用 这 些 图 数 库 能 够 提 霹 开发 者 的 效率 。 大 多 数 情 
况 下 ， 系 统 内 管 的 函数 库 都 能 够 满足 开发 需要 ， 当 然 恋 者 了 可 以 继续 完善 这 些 函 数 ， 让 其 更 
加 适合 项 目 和 需求。 本草 将 介绍 内 置 的 快捷 函数 亩 、 编 详 函 数 、 扩 展 函 数 等 。 


9.2.1 快捷 方法 


快捷 方法 实际 上 是 一 种 系统 函数 ， 之 所 以 为 称 为 方法 是 因为 此 类 函数 通 当 用 于 快速 实例 
化 系统 核心 类 库 ， 类 库 中 的 方法 通常 直接 在 快捷 函数 中 调用 即 可 。 所 以 官方 手册 习惯 把 快捷 
图 数 称 为 快捷 方法 。 系 统 内 症 的 快捷 方法 如 表 9-5 所 示 。 
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表 9-5 系统 快捷 函数 库 


表现 形式 


A A(name, app='@') 实例 化 Action, name 表示 Actin 名 称 app 表示 项 目 名 ， 默 认 


是 当前 项 目 
B B(name) 行为 调用 
C TE EH 获取 或 设置 配置 文件 数据 。 当 value 为 null 时 处 于 获取 模式 ， 
Een g 否则 为 设置 模式 
D Dname=",app=") 实例 化 自 定 义 模型 
读 取 或 删除 缓存 文件 。 当 为 空 时 处 于 删除 缓存 模式 ， 否 
G G($start,$end=",$dec=4) 以 微 秒 的 单位 统计 代码 执行 时 间 
获取 或 设置 多 语言 配置 。 当 value 为 空 时 处 于 获取 模型 ， 和 否则 
L Lname=null,value=nul]) 为 设置 模型 
M M(name=",class='"Model') 实例 化 数据 表 (Model 基础 模型 ) 
N N($key, $step=0) 设置 和 获取 统计 数据 
Re en 实例 化 当前 项 目 控 制 句 及 路 项 目 控制 器 动作 。 人 参数 module 为 控 
EE 制 器 名 称 ，action 为 动作 名 称 ，app 为 项 目 名 称 。 默 认为 当前 项 目 
获取 或 设置 缓存 。 与 F 函数 不 同 ，S 函数 能 够 获取 到 任何 形式 
S S(name,value=",expire=",type=") 的 缓存 数据 ， 包 括 Memcache, X-Cache 等 。value 为 空 时 处 于 获 
取 模 式 ， 否 则 为 设置 模式 
U U (url,params,redirect=false,suffix=true) 根据 当前 URL 配置 生成 URL 地 址 并 支持 跳 转 
W(name,data=array(),return=false) 输出 Widget 


快捷 方法 的 使 用 在 前 面 的 章节 内 容 中 已 经 多 次 所 及 。 例 如 M 函数 、D KASE, EEA 
再 重 述 ， 其 他 快捷 函数 在 以 后 的 革 市 内 容 中 将 会 有 所 提 及 ， 例 如 S 函数 、F 函数 等 。 


9.2.2 ”基础 函数 库 


系统 基础 函数 库 是 确保 系统 能 够 正常 运行 的 关键 ， 当 然 在 项 目 中 调用 系统 的 基础 函数 也 
是 被 允许 的 ， 接 下 来 将 分 别 介绍 儿 个 内 置 的 高 效 基础 函数 。 

1. file exists case (检查 文件 ) 

file_exists_case 函数 是 is_file 的 封装 函数 。is_file 是 PHP 内 置 的 一 个 基础 函数 ， 该 函数 
用 于 检测 指定 的 文件 是 否 存在 ， 如 有 存在 则 返回 true 否则 返回 falses {H7 is_file 函数 在 检测 
文件 时 是 不 区 分 大 小 号 的 ，file_exists_case KÆ is_file ëtt Tä, dk VIII 
区 分 大 小 写 。file_exists_case 格式 如 下 所 示 。 

file_exists_case($filename) 


参数 filename 表示 文件 的 路 径 〈 从 入 口 文件 开始 算 起 )。 使 用 方式 如 以 下 代码 所 示 。 


EE? 


echo file exists_case("./Public/Uploads/t.jpg"); 
} 
需要 注意 的 是 ，fe_exists_case 只 能 实现 在 Windows 系统 下 区 分 大 小 写 ， 在 非 Windows 下 
am, 5 fle_exists_case 函数 相似 功能 的 还 有 file_exists 函数 ， 该 函数 是 PHP 内 置 的 一 个 
标准 函数 ， 但 据 公 开 的 资料 显示 file_exists 与 is_file 性 能 相差 10 倍 以 上 Oe Die 性 能 较 高 )。 
2. reouire Cache (包含 文件 ) 
require_cache 函数 是 require_once 的 封 狼 。require_once 是 一 个 用 于 包含 文件 的 PHP H 
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础 水 数 ， 该 函数 是 PHP 5.x 新 增加 的 ， 用 于 避免 重复 的 include 引用 导致 程序 骨 涡 。 
require_once 在 引入 文件 时 ， 对 环境 变量 的 检测 是 不 严谨 的 。 最 典型 的 束 是 在 Windows 系统 
下 ，require_once 不 检测 文件 大 小 号， 导致 程序 移植 到 Linux 等 系统 时 运行 出 错 。 
require_cache KZH file_exists_case 冰 数 来 检测 文件 ， 能 够 避免 因 大 写 小 造成 的 错误 。 
require_cache 函数 格式 如 下 上 所 示 。 

require_cache($filename) 


参数 filename 表示 文件 的 路 径 《 从 入 口 文 件 开始 算 起 )。 如 以 下 代码 所 示 。 


public function index(){ 
require_cache("./Public/head.html") ; 

| 

3. Import (导入 类 ) 

import 函数 用 于 导入 系统 基础 类 库 、 应 用 类 库 的 党 用 函数 。import 参考 了 Java 编程 中 的 
import 文件 导入 机 制 ， 并 且 也 使 用 命名 空间 的 方式 进行 导入 。 所 谓 的 命名 空间 ， 事 实 上 是 相 
ITA Dm en, 4 Java 或 者 PHP 内 置 的 命名 空间 不 是 一 回 事 。import 命名 空间 导入 
格式 下 所 示 。 

import($class, $baseUrl = ", $ext='.class.php') 

其 中 clas 参数 表示 类 名 ， 类 名 必须 要 与 文件 名 (不 带 .class.php ARO 一 致 ， 这 是 
ThinkPHP 的 文件 命名 规范 ， baseUrl 参数 表示 基础 类 库 ， 即 引入 的 类 文件 路 径 从 该 目录 开始 
DH, ext 参数 表示 类 文件 后 缀 名 。 这 里 需要 重点 理解 的 是 class 参数 值 ， 该 参数 值 使 用 命名 
罕 间 的 方式 来 表示 ， 如 以 下 代码 所 示 。 

import ("Think.Util.Session"); 

其 中 Think 是 标识 符 ，Util 表示 命名 空间 ，S$ession KRŠE. PRD.” 
“/” 相 等 。 在 import 国 数 中 ， 标 识 符 是 一 个 很 重要 的 概念 ， 只 有 系统 能 够 兴 认 的 标识 符 ， 访 
基础 类 才能 被 导入 。 人 否则 系统 将 会 答 试 导入 当前 项 目下 的 类 库 。 系 统一 共 提 供 了 4 个 标识 符 
写 ， 分别 代表 不 同位 置 的 类 库 ， 如 表 9-6 所 示 。 


表 9-6 import 函数 导入 类 标识 符 


bk R 符 定位 路 径 说 明 
ORG /ThinkPHP/Extend/Library/ORG/ 扩展 网 络 处 理 类 库 
Com /ThinkPHP/Extend/Library/Com/ 扩展 接口 类 库 
@ /AppName/ Lib/ 当前 项 目 目录 下 的 Lib 目录 


上 述 人 代码 的 最 终 路 和 从 为 /ThinkPHP/Lib/Util/Session.class.php 。 假设 将 
import("Think.Util.Session") 改 为 import("Page.Util.Session") ， 那么 路 径 将 变 为 
/home/../Page/Lib/Util/Session.class.php。 这 是 因为 Page 并 非 标识 符 ， 当 系统 过 到 非 标识 符 
时 ， 会 将 路 径 定 位 到 当前 项 目 根 目录 下 。 同 样 ， 修 改 了 参数 baseUrl 系统 将 首先 定位 系统 根 
目录 《与 入 口 文件 平 级 )， 然 后 定位 到 baseUrl 指定 的 目录 下 。 需 要 注意 的 是 baseUrl 参数 值 
不 能 同时 与 “@ ”标识 符 同 在 。 

ext 参数 用 于 导入 第 三 方 扩 展 是 非常 有 用 的 ， 因 为 它 可 以 修改 默认 的 .class.php 类 文件 后 
级 名 。 例 如 有 一 个 第 三 方 的 类 ， 命 名 为 Page.php， 如 果 继 续 使 用 默认 的 导入 方式 显然 是 失败 


国 国 243 


PHP MVC 开发 实战 


的 ， 此 时 可 以 通过 下 述 两 种 方式 实现 类 库 的 导入 。 
import (Y0. Ucil, Page, YY, Y pap”) g 
mo ee (VA, UEL a PAGS e 


上 述 代码 导入 结果 都 是 相同 的 。 最 终 的 文件 路 径 为 ,home/Lib/Util/Page.php。 前 者 通过 修 
改 ext 参数 实现 ; 后 者 通过 使 用 “#” 稚 代 符号 实现 。“#” 蔡 代 符 号 是 系统 用 于 代替 “.” 文 
件 后 级 符 写 的 ， 这 样 束 可 以 避免 因 命 名 空间 与 非 .class.php 后 级 名 文件 造成 的 冲突 。 

虽然 使 用 import 能 够 实现 导入 第 三 方 类 库 ， 但 系统 同时 提供 了 vendor 函数 ， 该 函数 在 
导入 第 三 方 类 库 时 更 加 规范 与 便捷 。 

4. vendor( 导 入 第 三 方 类 ) 

前 面 提 人 到 过 ， 使 用 import 可 以 导入 任意 类 文件 。 事实 上 vendor 函数 也 只 是 对 import Gë 
数 的 封装 ， 这 种 封 闻 是 有 针对 性 的 ， 所 以 操作 起 来 更 加 简单 。 为 了 便于 文件 的 管理 ， 系 统 的 
第 三 方 扩展 包 都 存放 于 ThinkPHP/Extend/Vendor 目录 ，vendor Pë Zär ot AZEM ANH F 
的 ， 所 以 建议 所 有 第 三 方 类 库 或 扩展 包 都 放置 于 该 目录 下 ， 以 便 进 行 统一 管理 。ThinkPHP 
默认 情况 下 已 经 提供 了 多 个 扩展 包 ， 如 下 上 所 示 。 


EaseTemplate/ 


phpRPC/ 
SmartTemplate/ 
Smarty/ 
TemplateLite/ 
Zend/ 


假设 现在 需要 导入 Smarty 扩展 包 ， 使 用 vendor 函数 束 能 很 方便 地 导入 ， 如 以 下 代码 所 示 。 


public function index (){ 


Vendor ('Smarty .Smarty'); 


} 

Vendor KASZ import 函数 参数 一 样 ， 同 时 也 采用 命名 空间 方式 导入 文件 。 上 述 代 
码 最 终 得 到 的 路 径 为 ./ThinkPHP/Extend/VendorvSmarty/Smarty.php。 

5. load ($A RHD 

import 和 vendor 函数 在 导入 文件 时 都 遵循 命名 空间 的 方式 ，load 函数 也 用 于 导入 文件 ， 
但 与 前 面 介绍 的 函数 不 同 ，load 本 数 不 遵循 命名 空间 方式 ， 并 且 后 缀 名 默认 不 再 
是 .class.phpp。 从 这 个 意义 上 来 讲 ，load 函数 用 于 导入 扩展 函数 文件 是 最 合适 的 了 。l1oad RZ 
格式 如 下 所 示 。 

load($name, $baseUrl=", $ext='.php') 

其 中 参数 name 表示 文件 名 称 〈( 不 帘 后 级 名 ); baseUrl 为 基 上 有 目录， 默认 为 
/ThinkPHP/Extend/Function/; ext 表示 文件 后 级 名 。 

如 果 要 导入 的 文件 位 于 当前 项 目下 ， 那 么 可 以 在 baseUrl 参数 中 进行 指定 ， 也 可 以 使 用 
“@” 指 定 ， 如 以 下 代码 所 示 。 


public function index (){ 


Loa (WA tsen") ep 


} 
ERRIRE AN /home/Common/user.php. EH "e" 5 A ra ou 
H Common Hx, Common 正 是 用 于 存放 日 定义 函数 的 目录 。 最 后 导入 对 应 的 文件 名 。 
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6. strip_whitespace (REZA) 

strip_whitespace 函数 可 以 对 输入 的 数据 进行 优化 和 过 小 。 去 除 空 白 在 PHP 中 有 很 多 种 方法 
可 以 实现 ， 例 如 使 用 内 置 的 函数 、 正 则 表达 式 替 换 等 。 但 使 用 strip_whitespace 函数 不 仅 可 以 去 
除 空白 ， 还 可 以 去 除 代 码 中 的 注释 ， 最 后 将 结 末 合 并 为 一 行 数 据 ， 极 大 地 压缩 代 但 质量 。 系 统 
编译 (Runtime) 代码 时 正 是 使 用 strip_whitespac 函数 实现 的 ， 开 发 人 员 也 可 以 直接 调用 。 

7. mk dir (创建 目录 ) 

PHP AEH mk_dir 函数 能 够 方便 地 创建 目录 ，ThinkPHP 封装 后 的 mk_dir 函数 用 于 实现 在 
创建 前 对 指定 目录 进行 检测 。 如 果 目 录 存 在 则 创建 ， 人 否则 将 直接 返回 false， 并 且 不 会 报错 。 


9.2.3 ”扩展 函数 库 


系统 扩展 图 数 库 能 够 增强 系统 的 基础 特性 ， 在 使 用 时 需要 使 用 LoadCextend) 快 捷 方 法 引 
入 扩展 函数 库 。 下 面 将 分 别 对 常用 的 扩展 函数 进行 讲解 。 

1. auto charset (自动 编码 ) 

auto_charset 基础 函数 封 厂 了 iconv 函数 ， 能 够 对 gbk, gb232 或 者 utf-8 等 数据 进行 相互 
MI, auto charset 是 系统 扩展 函数 ， 使 用 时 需要 额外 引入 ， 格 式 如 下 所 未 。 

auto_charset($fContents, $from='gbk', $to='utf-8') 

其 中 参数 {Contents 表示 需要 进行 编码 的 数据 ，from 参数 表示 当前 数据 编码 类 型 ，to 参 
数 表示 需要 转换 的 目标 编码 ， 常 用 参数 值 有 gbk、gb2312、utf-8 等 。 下 面 将 使 用 一 个 例子 示 
70 auto_charset 国 数 的 使 用 ， 如 以 下 代码 所 示 。 


public function index (){ 
Load ('extend'); 
$str=auto_charset (" 编 公 测试 ", "utf-8", "gbk"); 
header ('Content-Type:text/html; charset=gbk'); 
exit ($str); 


} 

在 ThinkPHP 中 ， 文 件 编码 必须 使 用 UTF8， 所 以 当前 页 面 中 的 数据 也 是 utf-8 编码 。 使 
用 auto_charset 函数 将 utf-8 转换 为 gbk 后 ， 如 末 不 显 式 指定 charset 为 gbk， 浏 览 器 仍然 使 用 
utf-8 模式 来 处 理 当前 页 面 输出 ， 这 样 的 后 果 显 然 是 错误 的 (中 文 出 现 乱 人 码 )。 

2. is_utf8 检测 是 否 UTF8 编码 ) 

由 于 系统 只 文 持 UTF8 编码 的 文件 ， 所 以 当 数 据 不 是 UTF8 编码 时 ， 将 不 能 够 正确 处 
理 。 这 时 可 以 使 用 is_utf8 函数 对 数据 进行 检测 ， 如 果 不 是 utf-8 编码 时 ， 使 用 前 和 耐 介绍 过 的 
auto_charset 国 数 转换 。is_utf8 函数 格式 如 下 所 示 。 

is_utf8($string) 

下 面 通过 一 个 示例 演示 is_utf8 扩展 水 数 的 使 用 ， 代 人 码 如 下 所 示 。 


public function index (){ 
Load ('extend'); 
$str=auto_charset ("编码 测试 ", "utf-8", "gbk"); 
if (is utf8($str))t{ 
ooer aneo chareet i ek gutt-Bni: `) 
header ('Content-Type:text/html; charset=utf-8'); 
Ers 
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3. msubstr〈 中 文 截取 ) 

PHP 内 置 了 众多 字符 串 处 理 的 图 数 ， 对 字符 串 进 行规 取 的 有 substr、mb_substr、 
mb_strcut, str_replace 和 等， 当然 还 有 强大 的 正则 表达 式 。 除 了 正则 表达 式 之 外 ，PHP AEN 
字符 截取 函数 对 中 文 的 文 持 都 是 不 完善 的 ，msubstr 利用 了 PHP 正则 及 mb_substr 函数 ， 实 
现 了 对 中 文字 符 串 的 精确 截取 。msubstr 函数 格式 如 下 所 示 。 

msubstr($str, $start=0, $length, $charset="utf-8", $suffix=true) 

其 中 参数 str 表示 需要 鹤 取 的 字符 串 ， 文 持 瑞 文 、 数 字 、 中 文 〈 和 体 或 驼 体 ); 参数 start 
表示 截取 的 开始 位 置 ， length 表示 稚 取 的 最 后 一 个 字符 位 置 ， 默 认为 不 限制 ， 即 一 直人 到 学 符 
串 最 后 一 个 学 符 ; charset 表示 截取 的 字符 串 编 权 ; 参数 suffis 为 mue 时 把 截取 简 余 部 分 用 
“...” 代 和 罕 。 下 面 通 过 示例 演示 msubstr 函数 的 使 用 ， 代 码 如 下 所 示 。 


public function index (){ 
Load ('extend'); 
$str="mb_substr MÁT — ee substr () 操作 基础 上 的 字符 数 。 
从 str 的 开始 位 置 计算 。 第 一 个 字符 的 位 置 为 0。 第 二 个 字符 的 位 置 是 1， 依 此 类 推 。"， 
echo EU5SEE SSEE 12,23)} 


} 

最 终 将 会 得 到 “一 个 多 字 和 安全 的 substrO 操 作 基础 上 的 宇 …”， 读 者 可 以 在 此 基础 上 继 
续 完 善 msubstr 函数 。 例 如 修改 截取 后 缀 字符 “.…” 等 。 

4. rand_string〔 随 机 生成 密码 ) 

PHP AHH rand 函数 可 以 方便 地 随机 生成 数 季 和 字 从 串 。rand_string 对 rand 函数 进行 了 
封 狼 ， 不 仪 实现 了 纯 数字 随机 、 了 字母 随机 、 数 学 与 字母 混合 随机 ， 还 实现 了 中 文学 符 随机 、 
中 文 与 英文 混合 随机 等 。rand_string 通 名 用 于 目 动 生成 密 合 ， 但 也 可 以 用 于 需要 目 动 生成 字 
和 从 的 场景 ， 例 如 验证 码 、 安 全 问答 话题 等 。rand_string 函数 格式 如 下 。 

rand_string($len=6, $type=",$addChars=") 

其 中 参数 len 表示 随机 生成 学 符 串 的 位 数 ， 参 数 type 表示 随机 字符 串 的 类 型 ， 共 文 持 6 
种 类 型 ，addChars 表示 额外 字符 〈 即 后 级 )。 下 和 面 对 过 示例 演示 rand_string 函数 的 使 用 ， 代 
但 如 下 所 示 。 


public function index (){ 
Load ('extend'); 


Şstr=rand_string (6,3); 


echo S$str; 


} 
在 实际 应 用 开发 中 ， 通 常 需要 修改 参数 type， 改 变 该 参数 的 值 ， 将 直接 影响 到 随机 字符 
DIE d. type 参数 有 效 值 由 0 一 5 组 成 ， 如 表 9-7 所 示 。 


表 9-7 type 参数 值 


type 值 字符 有 效 范围 生成 样式 
0 A-Z, a-z rRnTQL 
0-9 722199 
2 A-Z EFAHGV 
3 a-Z cabikw 
4 SE SS m 2 2. 
5 除 参 数值 4 之 外 的 有 效 字 符 xAbZv6 
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5. build_count_rand( 批 量 生成 随机 字符 串 ) 

rand_string 苹 数 一 次 只 能 生成 一 串 字 人 符 串 ，build_count_ rond 函数 是 基于 rand_string Pë Sr 
的 ， 但 build_count rand 函数 实现 了 批量 生成 随机 字符 串 ， 并 将 结果 保存 到 数组 中 。 
build_count rand 函数 格式 如 下 所 示 。 

build_count_ rand ($number,$length=4,$mode=1) 

参数 number 表示 随机 批量 生成 的 字符 串 数量 ， 参 数 length 表示 字符 串 的 长 度 ，model 
参数 与 rand_string 函数 中 的 type 参数 相同 。build_count_rand 函数 使 用 非常 简单 ， 如 以 下 代 
Elo 


public function index (){ 
Load ('extend'); 
$str=build_count_rand (6, 6,3); 
dump ($str); 


} 

build_count rand 函数 返回 的 结果 是 关联 数组 ， 在 实际 应 用 开发 中 ， 可 以 将 结果 循环 皇 
入 数据 库 等 。build_count rand 函数 通常 用 于 批量 生成 优惠 券 、 虚 拟 金 币 等 。 

6. byte format 〈 字 节 格 式 化 ) 

byte_format 一 个 格式 化 计算 机 容量 单位 的 函数 。 默 认 情 况 下 ， 计 算 机 都 是 以 byte (B) 
为 单位 的 ， 但 对 用 户 而 言 显然 是 不 友好 的 ， 一 个 普通 的 网 络 文件 少 则 儿 十 KB; 大 的 几 十 
MB 甚至 儿 百 MB。 这 种 情况 下 束 需 要 对 网 络 文件 容量 单位 进行 格式 化 。byte_format Pë Zär nl 
以 实现 对 byte 自动 格式 化 为 B、KB、MB、GB、TB、PB， 对 一 般 的 网 站 而 言 ， 最 常见 的 为 
KB 及 MB. byte_format 函数 格式 如 下 。 

byte_format($size, $dec=2) 

其 中 参数 size 表示 要 转换 byte 数据 大 小 ，1024byte 约 等 于 1KB; 参数 dec 表示 四 舍 五 
入 位 数 。 下 面 通过 示例 演示 byte_format 函数 的 使 用 ， 代 但 如 下 所 示 。 


public function index (){ 
Load ('extend'); 
echo byte_format ("1073741824",1); 


en 


上 述 代 但 结果 将 显示 “1 GB”。 需 要 注意 的 是 byte_format 在 Windows 平台 与 Linux 等 平 
台 格 式 化 结果 有 所 差异 (Linux 平台 以 1000byte 为 基础 运算 单位 )。byte_format RAOK zé 
于 上 传 文件 检测 ， 或 者 统计 用 户 使 用 空间 大 小 等 。 

7. highlight_code (代码 加 亮 ) 

在 一 些 需 要 原生 代码 演示 的 文章 系统 中 ， 经 党 需要 航 入 高 之 代码 ， 并 且 和 需要 保持 代码 原 
有 的 格式 、 衬 体 颜 色 、 行 号 等 。 传 统 的 做 法 是 先 将 代码 使 用 htmlspecialchars 转 义 ， 然 后 保存 
到 数据 库 中 ， 取 出 时 再 使 用 htmlspecialchars_decode 函数 反 转 义 ， 最 后 使 用 HTML 标记 
<pre> 原 格式 输出 (还 需要 结合 正则 和 巷 换 )。 

使 用 highlight_code 函数 输出 原 代码 格式 ， 将 和 变 得 人 简单， 该 函数 所 接收 的 代码 数据 即 为 
最 终 呈 现 的 代码 格式 。 不 仅 如 此 ，highlight_code 函数 还 支持 直接 传 入 文件 路 径 ， 返 回 该 文件 
原生 代码 。highlight_code 函数 格式 如 下 所 示 。 

highlight_code($str,$show=false) 

参数 str 表示 原生 格式 代 公 ;参数 show Kanem HI, N true 时 返回 值 ， 不 输出 。 
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下 面 通过 示例 代码 演示 highlight_code 函数 的 使 用 ， 代 码 如 下 所 示 。 
public function index (){ 
Load ('extend'); 
Şstr=highlight_code ("./index.php"); 
header ('Content-Type:text/html; charset=utf-8'); 


echo $str; 


} 
上 述 代 码 最 终 运 行 结果 如 图 9-6 所 示 。 


<?php 
define("THINK_PATH","./ThinkPHP/"); 
define("APP_PATH","./home/"); 

define" APF_NAME", " Home"); 
define("APP_DEBUG", true); 

. require_once(THINK_PATH."ThinkPHF.php"); 


Fa 


本 A A d dd nm 


图 9-6 highlight_code 高 亮 代码 效果 


9.3 ThinkPHP 多 语言 支持 


多 语言 环境 是 每 个 国际 化 网 站 都 需要 的 功能 ， 在 传统 的 PHP FEP, FRANH m 
要 使 用 多 套 模板 、 多 种 类 库 实 现 多 语言 文 持 。 在 主流 的 MVC 框架 中 ， 几 乎 都 提供 了 多 语言 
LFF, ThinkPHP 同样 也 对 多 语言 提供 了 完善 的 文 持 ， 接 下 来 将 全 面 介绍 。 


9.3.1 部 署 多 语言 

使 用 多 语言 功能 ， 能 够 有 效 地 降低 开发 国际 化 网 站 的 难度 。 默 认 情 况 下 ， 系 统 是 关闭 多 
语言 功能 的 ， 所 以 也 不 需要 语言 包 。 项 目的 语言 包 默 认 存 放 于 项 目 Lang 目录 下 ， 一 个 典型 
的 Lang 目录 及 文件 结构 如 图 9-7 所 示 。 


Lang/ 
d UV E e ege 
index .php pa >》 Index 控 制 避 
User, Dim iiio > User 控 制 器 
M(ticle Dh a 》 Article 控 制 器 
common.php E 》 公共 语言 
zt w/e BEES 
index.php ................. 》 Index 控 制 器 
user POD cerraron: > User 控 制 器 
artcle .php .ee 》 Article 控 制 器 
common.php ee y Am 


图 9-7 项 目 语言 包 目录 及 文件 结构 
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在 实际 应 用 开发 中 ， 如 条 需要 更 多 语言 文 持 ， 根 据 上 述 目 录 及 文件 结构 添加 即 可 ， 但 需 
要 确保 各 语言 包 的 结构 是 一 致 的 。 例 如 增加 en-us 包 ， 该 包 的 目录 及 文件 结构 必须 与 其 zh-cn 
或 zh-tw 包 结 构 相 同 。 公 共 文 件 common.php 不 是 必需 的 ， 但 为 了 更 加 容易 理解 ， 这 里 将 保 
留 该 文件 : 

通 第 common.php 语言 文件 用 于 配置 当前 语言 包公 共 部 分 的 语言 数据 ， 例 如 目 定义 也 
数 信 息 返 回 、 日 志 操 作 等 。 如 果 项 目 非 常 小 ， 甚 至 只 需要 使 用 common.php 文件 作为 语言 
包 即 可 。 


9.3.2 ”实现 多 语言 


部 署 好 语言 包 后 ， 驶 需要 对 语言 包 中 的 配置 文件 进行 配置 了 。 这 里 需要 说 明 的 是 ， 
ThinkPHP 多 语言 并 非 指 系统 目 动 实现 语言 翻译 ， 尽 管 系统 内 置 了 目 动 定位 语言 的 功能 ， 
但 不 能 实现 目 动 翻译 功能 ， 所 谓 的 语言 包 文 件 需要 开发 人 员 配 合 翻译 人 员 修 改 配置 值 实 
现 。 假 设 需要 对 Index 控制 右 配 置 徐 体 与 党 体 中 文 ， 那 么 zh-cn/index.php 代码 如 下 所 
ZN o 


<?php 

return array 人 
// 简 单 中 文 
Beer 首页 对 
"title"=>" 标 题 "， 
Eet 
"category si" "mAn 17", 
vadda user =- HS IER Pin. 
tadda einet > N 
"submit_btn txtn=>" 提 交 "， 
"aqdqd_errorn=>" 文 章 添 加 失败 "， 
tadda success" Sw KO, 

); 


对 应 的 繁体 中 文 语言 配置 文件 zh-tw/index.php 代码 如 下 所 示 。 
<?php 


return array 人 
/ /繁体 中 文 
"page title= ml 
"title"=>" 标 题 "， 
ee Se 
人 
"add user tes HS JIH", 
"add_time"=>" 添 加 时 间 "， 
"submit_btn txtn=>" 提 交 "， 
Beet 
"add success les HME ES DIEN Din, 


1: 


和 其 他 配置 文件 一 样 ， 语 言 配置 文件 也 是 使 用 关联 数组 来 配置 的 。 控 制 器 及 动作 中 的 语 
言 配 置 文件 优先 级 大 于 common.php 文件 。 
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9.3.3 ”多 语言 与 客 忆 痛 

通过 前 面 的 设置 ， 现 在 系统 已 经 具备 了 多 语言 编程 的 条 件 ， 但 现在 系统 还 不 能 够 检测 到 
多 语言 配置 的 存在 ， 因 为 还 需要 在 配置 文件 中 开局 多 语言 文 持 〈 默 认 是 关闭 的 )， 然 后 还 需 
要 配置 系统 行为 ， 让 系统 行为 载 入 多 语言 文件 。 

1. 开局 多 语言 环境 

与 其 他 类 库 一 样 ， 开 启 多 语言 只 需要 在 配置 文件 中 开启 LANG SWITCH ON 选项 即 
可 ， 如 以 下 代码 所 示 。 


<?php 

return array 人 
'LANG_SWITCH_ON'=>true，// 是 否 开 启 多 语言 
'LANG_AUTO_DETECT'=>true，// 是 否 自动 检测 语言 环境 
'DEFAULT_LANG'=>'zh-tw'，// 默 认 语 言 包 
// 省 略 配置 项 .. 


) 


Pe 


完成 上 述 配 置 项 后 ， 还 需要 在 项 目 Conf 目录 下 创建 tags.php 文件 ， 然 后 在 该 文件 中 开 
D CheckLang 行为 ， 如 以 下 代码 所 示 。 


<?php 


return array 人 

/77 配置 系统 行为 

'app_begin'=>array ('CheckLang') 
); 


通过 前 面 的 配置 ， 现 在 就 可 以 使 用 多 语言 来 编程 了 。 

2. 使 用 多 语言 环境 

在 MVC 开发 中 ， 主 要 有 两 种 形式 的 编程 环境 需要 使 用 多 语言 。 一 种 是 在 视图 模板 中 使 
用 ; 另外 一 种 是 在 控制 器 动作 、 模 型 中 使 用 《〈 即 在 后 台中 使 用 )， 下 面 分 别 介绍 。 

(1) 在 视图 模板 中 使 用 多 语言 

视图 是 用 户 首先 接触 也 是 接触 最 多 的 MVC 组 件 ， 所 以 应 用 多 语言 也 主要 应 用 于 视图 模 
板 。 为 了 方便 演示 ， 这 里 将 index 动作 对 应 的 index.html 模板 改 为 国际 化 多 语言 模板 ， 如 以 
下 代码 所 示 。 


<style> 


.main li{ 
list-style s nonez 
margin:8px; 


} 


</style> 

<div class="main"> 

<form action=" URL /post" method="post"> 
<ul> 


<1i><!--{$Think.1lang.title}—-—>: 


<input type="text" size="60" name="title"/> 


</1i> 
<l1i><!--{$Think.1lang.category}—-->: 
<label> 


<select name="select" id="select"> 
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<option value="0"><!-—-{$Think.lang.category_sl}--></option> 
</select> 
</label> 
</I> 
<li><!-—-{$Think.lang.add_user}-—>: 
<input name="add_user" type="text" id="add user" size="22" /> 
过 /下 工交 
<li><!-—-{$Think.lang.add_time}--—>: 
<input name="add_time" type="text" id="add_time" size="22" value="<?php echo 
TME o 2> I> 
</li> 
<li> 
<label> 
<textarea name="content" id="content" cols="60" rows="8"></textarea> 
</label> 
</li> 
<li> 
<input type="submit" name="þutton" id="button" value="<!—{$Think.lang.submit_btn_txt}- 
一 >" /> 
SS 
SE 
</form> 
</div> 


FEI ANARA, BGLM ERKRATH AEE 
的 ， 而 是 来 自 于 语言 包 。 效 果 如 网 9-8 所 示 。 


HA English 


f> ES EA | fcr a s i 
欢迎 光临 我 的 博客 
首页 | FABE | 搜索 中 心 | #204 
相册 D 
分 类 : 证 选手 一 个 分 类 e (E 
Sub : LL E 
| 网 站 公告 
Se) : 1351521545 
| 习 咸 


首页 | 联系 方式 | 甘于 本 站 | 2AE | 订阅 


图 9-8 简体 中 文 首页 表单 


表面 提 到 过 ， 在 默认 情况 下 系统 会 自动 识别 当前 的 浏览 器 语言 环境 ， 并 调用 相应 的 语言 
包 。 开 发 人 员 也 可 以 传 入 GET 参数 1 (或 L)， 实 现 手动 定位 语言 包 ， 例 如 访问 
http:/Wtp.localhosty?1=zh-tw， 表 单 中 的 文字 将 转换 为 简体 中 文 模 式 ， 如 网 9-9 所 示 。 

需要 注意 的 是 ， 一 旦 选 定 了 指定 语言 包 ， 系 统 将 会 记录 ， 下 次 再 访问 该 页 面 时 ， 系 统 将 
目 动 巨 配 上 次 选择 的 语言 包 。 读 者 可 以 在 此 基础 上 继续 完善 ， 实 现 全 页 面 的 多 语言 转换 ， 例 


EA 
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如 将 分 类 下 拉 列 表 使 用 多 语言 包 实现 。 


欢迎 尖 | 者 我 的 博客 DD ig 


首页 | WAHE | ERTU | Ss 


标题 ; me 

分 类 : i " am 

AMHER : 

E RADAT ， 欢 迎 来 访问 我 的 


首页 | 联系 方式 | ee EE, Mäer 


图 9-9 索 体 中 文 首页 表单 


(2) 在 后 合 中 使 用 多 语言 

在 后 台中 使 用 多 语言 通常 用 于 系统 提示 信息 、 日 志 记 录 、 表 单 验证 等 。 前 面 的 章节 中 已 
经 介绍 过 工 函数 ， 该 函数 就 是 用 于 在 后 从 获取 或 设置 语言 包 的 快捷 函数 。 下 面 将 通过 代码 演 
示 工 函数 的 实际 使 用 ， 代 码 如 下 所 示 。 


public function add()t 
SUser=D ("Article"); 
if (!$User->create()){ 
throw exception (L (tadda error) ; 


}else{ 
$this->success (L("add_success")); 


} 


94 客户 站 


ThinkPHP 内 置 了 许多 针对 客户 端的 类 库 ， 使 用 这 些 类 库 更 加 符合 MVC Gd, Jo 
时 也 能 够 有 效 地 提高 开发 效率 。 本 节 将 对 常见 及 重要 的 Session、Cookie 类 库 进 行 介 绍 。 


9.4.1 封闭 的 Session 


网 站 开发 都 离 不 开 临 时 会 话 〈S$ession) 功能 。 这 些 数 据 是 临时 性 的 ， 但 也 是 非常 重要 
的 ， 例 如 记录 登录 用 户 数 据 、 用 户 爱 好 ， 统 计 网 站 数据 每 都 离 不 开 Session。 在 PHP 开发 中 
使 用 Session 是 非常 简单 的 ， 开 发 人 员 只 需要 使 用 session_startO 函 数 开 局 即 可 。 虽 然 如 此 ， 
但 在 实际 应 用 开发 中 通常 是 不 能 满足 要 求 的 ，ThinkPHP 一 共 内 置 了 两 种 Session 封装 操作 ， 
分 别 为 Session 函数 及 Session 扩展 类 库 。 这 两 种 封 疤 操 作 都 是 融 效 、 灵 活 的 。 下 和 面 首先 介绍 
Session 函数 。 
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1. Session 函数 

Session FÉIZ ThinkPHP 3.x 提供 的 一 个 功能 完善 的 Session 管理 函数 ， 它 封装 了 所 有 
Session 常规 操作 ， 例 如 session start 初始 化 、session id 增 、 删 、 改 、 查 等 。Session GL 
供 了 Session Hander 驰 动 扩展 功能 ， 开 发 人 员 可 以 轻易 地 利用 该 驱动 扩展 ， 编 瑟 高 效 的 Session 
储存 机 制 〈 例 如 存放 到 内 存 数据 库 、Memcache、NoSQL E). Session 函数 格式 如 下 所 示 。 

session($name,$value=") 

其 中 参数 name 是 一 个 比较 特殊 的 变量 ， 一 般 情况 下 用 于 表示 Sessioon FIER CRH 
key); value 表示 key XARA, LIP. Session 函数 是 系统 内 置 的 核心 水 数 ， 使 用 时 不 
需要 额外 引入 ， 一 个 最 简单 的 Session 声明 如 以 下 代码 所 示 。 


public function index (){ 


session("goods","iPhone"); 


} 
Session 声明 之 后 ， 可 以 在 同 域 下 使 用 ， 如 以 下 代码 所 示 。 


public function testInaex() { 
echo session("gools") 


} 

可 以 使 用 dump($_SESSION) 函 数 输出 当前 域 下 的 所 有 Session。 参 数 name 是 一 个 特殊 
的 变量 ， 之 所 以 特殊 是 因为 该 变量 是 多 变 的 ， 当 变量 值 为 字符 串 时 ， 将 作为 Session 存放 
key; 当 值 为 关联 数组 时 ， 将 作为 Session 初始 化 的 环境 配置 。name 数组 的 配置 信息 如 表 9-8 
rz, 


表 9-8 参数 name 数组 值 配置 选项 


数 组 键 Ù 默 认 E 
id Session 存放 名 称 目 动 生成 
name Session 会 话 名 称 自动 生成 
path Session 存放 路 径 由 session.save_path 配置 信息 指定 
prefix Session Key 存放 前 级 A 
expire Session 过 期 时 间 由 session.gc_maxlifetime 配置 信息 指定 
domain Session 会 话 有 效 域名 当前 域名 〈 不 包括 子 域名 ) 
use Cookies Session 浏览 器 端 Cookie 名 称 自动 生成 
use frans gid 浏览 器 禁用 Cookie 时 传递 的 GET 变量 目 动 生 成 “Windows 无 效 ) 
type Session Hander 扩展 类 库 名 SS 


事实 上 ， 在 实际 应 用 开发 中 极 少 使 用 参数 name 来 初始 化 Session 环境 ， 因 为 该 参数 只 
针对 当前 Session 声明 生效 ， 如 果 在 其 他 地 方 继续 声明 Session， 所 有 配置 将 不 起 作用 ， 所 以 
更 通用 的 办 法 是 在 配置 文件 中 配置 ， 如 以 下 代码 所 示 。 

RON 

'SESSION PREFIX'=>'thk ',//Session 前 级 

'VAR_SESSION_ID'=>time (),//Session ID 

'SESSION_TYPE'=>'',//Session 驱动 类 库 ， 默 认为 文件 

"SESSION_OPITIONS '=>array ( 

"Parn" ss" t; 
'domain'=>'tp.localhost', 
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配置 项 SESSION_OPTIONS 定义 的 值 即 为 参数 name 的 值 。 初 始 化 完成 后 ， 就 可 以 实现 
对 Session 的 增 、 删 、 改 、 查 了 ， 如 表 9-9 所 示 。 


表 9-9 Session 常规 操作 


增加 session("goods","iPhone"); 增加 goods Session 项 ， 并 赋值 为 iPhone 
WE ` session('goods"mul); 删除 goodsSession 项 
修改 session("goods", "HTC"; Y goods Session TABOY “HTC” 
© Æ session(goods"); 获取 goodsSession 项 的 人 


前 面 提 到 过 ，Session 函数 文 持 hander KAP E, F PRK AENA hander 驱动 机 制 ， 
然后 结合 示例 代码 介绍 驱动 控制 的 实战 应 用 。 

PHP 内 置 的 Session 处 理 机 制 都 是 由 session_set save_handler 函数 来 完成 增 、 删 、 改 、 
AH, PH 允许 开发 人 员 重 载 该 函数 ， 这 就 意味 看 只 要 掌握 session_set_save_handler Eë Si DI 
人 使用， 就 可 以 将 Session 的 存放 环境 改变 为 PHP 所 文 持 的 环境 ， 例 如 大 型 数据 库 、Hadoop、 
NoSQL 等 ， 这 样 不 仅 说 来 性 能 的 提升 ， 还 能 实现 Session 路 网 站 应 用 ， 防 止 CC 攻击 等 。 
session set Save_handler 函数 参数 表现 形式 如 下 所 示 。 

session_set_save_handler(array(_ CLASS 'open'), 

array(__CLASS_, close'), 
array(__CLASS_, read'), 
array(__CLASS_, 'write'), 
array(__CLASS_, 'destroy'), 
array(_ CAS 'gc') 


) 

每 个 参数 由 一 个 关联 数组 组 成 。 数 组 中 的 “CLASS_ ”表示 操作 类 ， 对 应 的 值 即 为 
Session 操作 标识 ， 6 个 参数 表示 6 个 成 员 方 法 ， 分 别 为 open WIRE Session), close W 
Dr Session), read (获取 Session), write (Session 写 入 )、destroy 〈 全 部 销毁 Session), gc 
( 增 圾 回收 )。 根 据 这 些 方法 声明 ， 开 发 人 员 只 和 需要 在 对 应 的 类 中 实现 这 6 个 成 员 方 法 ， 即 可 
实现 高 级 的 Session 功能 定制 。 

ThinkPHP 内 置 的 Session 驱动 扩展 也 是 茹 于 session_set_save_handler 接口 冰 数 的 。 张 动 
类 库 默 认 存 放 于 ThinkPHP/Extend/Driver/Session 目录 ; 类 库 命 名 规则 为 “Session+ 类 库 名 ” 
(类 库 名 必须 首 字母 大 写 )， 该 名 称 即 为 SESSION_TYPE 配置 项 中 指定 的 名 称 ， 例 如 
SessionDb。 

接 下 来 将 以 系统 内 置 的 SessionDb 驱动 扩展 为 例 ， 详 细 介 绍 Session 数据 转 存 MySQL 数 
据 库 的 过 程 。 要 开局 Session 驱动 扩展 ， 首 先 需 要 在 配置 文件 中 配置 相应 选项 ， 如 以 下 代码 
所 示 。 


“SESSION AUTO STARI >true, // RH session star. O. AAA Ee 
'SESSION_PREFIX'=>'thk_',//Session 前 级 
I'VAR SESSION ID'=>time(),//Session ID 
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'SESSION_EXPIRE'=>36000, //session 过 期 时 间 
SESSION_TRABIE '=>'think_session'，// 存 放 session 的 数据 表 名 
'SESSION_TYPE'=>'Db', // 张 动 器 名 称 
SESSION_TABLE 指定 的 是 Session 存放 表 ，SQL 代码 如 下 所 示 。 
CREATE TABLE think session ( 

session_id varchar (255) NOT NULL, 

session_expire int (11) NOT NULL, 


session_data blob, 
UNIQUE KEY `session_id` (`session_id`) 


); 
当然 ， 读 者 也 可 以 使 用 其 他 数据 表 作为 Session 存放 表 ， 只 需要 确保 结构 如 图 9-10 所 示 
DI nl. 
FE 类 型 整理 
session_id varchart255) utf general_ci 


Session expire int{l1) 


Session datz varchart 220) uf general d 


图 9-10 Session 数据 存放 表 结 构 


通过 上 述 步 又 ， 现 在 再 使 用 Session 函数 操作 Session 时 ， 将 全 部 转 存 到 数据 表 ， 例 如 增 
加 一 条 Session 数据 ， 如 以 下 代码 所 示 。 


public function index (){ 


session ("username", Zënn: 


} 
存放 到 数据 表 中 的 Session 数据 如 图 9-11 所 示 。 


session id Session expire session data 


lnoec/kaBvgqqje8sesgpqpbuceé 1351650333 gools|s:8:"ttsdfsdf"; 
图 9-11 Session 数据 存放 表 


2. Session 类 
系统 提供 了 一 个 Session 扩展 类 ， 该 类 在 ThinkPHP 2.x 中 是 默认 的 Session 处 理 机 
制 。Session 类 只 是 对 PHP RAHI Session 操作 机 制 进行 了 封 疾 ， 让 其 更 加 适合 MVC 开 
发 。 在 ThinkPHP 3.x 中 使 用 时 ， 需 要 额外 手动 引入 该 类 。 下 面 将 结合 示例 代码 ， 演 示 
Sesson 类 的 使 用 。 
(1) 增加 Session 
public function index (){ 
import ("ORG.Util.Session"); 
Session: :set ("username", "F "); 
} 
(2) 删除 Session 


public function index (){ 
import ("ORG.Util.Session"); 
Session: :set ("username", null); 
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(3) 修改 Session 
public function index (){ 
Import (VORG. UTLI SESSLONY) p 


Session: :set ("username", "ceiba"); 


| 

(4) SEH Session 

public function index (){ 
import ("ORG.Util.Session"); 
echo Session: :get ("username"); 


| 
(5) 判断 Session 是 否 过 期 
public function index (){ 
import ("ORG.Util.Session"); 
if (Session:: isExpired ("username") ){ 
echo Session EAn H7 
} 


} 
(6) 删除 所 有 Session 


public function index (){ 
import ("ORG.Util.Session"); 
Session:: clear (); 


9.4.2 JAJ Cookie 


Cookie WWF R Pae Ur A, 5 Session ÑA, Cookie 也 是 用 于 存放 客户 
闹 临 时 数据 的 ， 不 同 之 处 在 于 Cookie 存放 于 客户 并 操作 系统 文件 夹 上 (由 浏览 器 指定 )， 而 
Session 存放 于 服务 器 指定 的 文件 夹 (或 由 Session 驱动 指定 存放 环境 )。 这 束 意 味 着 Cookie 
是 能 够 被 用 户 操作 ， 但 是 由 于 存放 于 客户 端 ， 服 务 堪 的 IO 压力 将 会 得 到 改善 〈 在 超大 流量 
的 网 站 中 尤其 突出 )。 所 以 一 般 情 况 下 如 果 网 站 的 安全 性 需求 比较 高 ， 需 要 使 用 Session; 如 
果 是 一 些 比较 开放 的 数据 ， 例 如 访客 统计 、 个 性 化 设置 等 可 以 使 用 Cookie, 

Cookie 的 操作 与 Session 相 类 似 ， 但 比 Session 简单 许多 。 同 样 ，ThinkPHP 也 提供 了 
Cookie KAH TAHEA Ym Cookie， 该 函数 格 式 如 下 所 示 。 

cookie($name, $value=", $option=null) 

其 中 参数 name 表示 Cookie 存放 名 称 〈 即 key)， 不 支持 数组 ; value 表示 Cookie 值 ; 
option 表示 Cookie 初始 化 环境 配置 参数 ， 该 参数 中 的 配置 参数 可 以 通过 配置 文件 统一 配置 。 
下 面 通 过 示例 代码， 演示 Cookie 函数 的 实际 应 用 。 

1. 增删 改 查 操作 

(1) 增加 Cookie 


public function index (){ 
Cookie ('ceiba', "F I 3600); 


} 

(2) 删除 Cookie 

public function index (){ 
Cookie ("cesenda' noll)s 


} 
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(3) 修改 Cookie 

public function index (){ 
cookie (ceiba, SFI); 

| 

(4) 获取 Cookie 

public function index (){ 
cookie ('ceiba'!); 


} 
(5) 删除 所 有 Cookie 


public function index (){ 
cookie (null); 


} 

(6) 删除 指定 前 组 的 Cookie 

public function index (){ 

COOkie Gawl il, warm Ne 

} 

对 于 查询 而 言 ， 不 管 Cookie 前 级 设置 如 何 ， 只 要 使 用 cookie 函数 来 获取 ， 系 统 都 
能 够 日 动 识 别 。Cookie 能 够 存放 包括 中 文 在 内 的 学 符 ， 但 长 度 需 要 控制 在 225 个 字符 
UN, 

2. Cookie 配置 

前 面 提 到 过 Cookie 函数 支持 参数 配置 ， 但 通常 情况 下 只 需要 在 配置 文件 中 配置 即 可 。 
Cookie 是 本 地 化 保存 机 制 ， 不 需要 驱动 扩展 ， 只 需要 指定 存放 路 径 、 有 效 期 、 作 用 域 即 可 ， 
如 以 下 代码 所 示 。 


'COOKIE EXPIRE'=>3600,//Cookie 有 效 期 
'COOKIE_DOMAIN'=>'tp.localhost', //Cookie 有 效 作用 域 
LCOOKEIE PATI, // Cookie TNA 

COOKIE PREETX tp Cooker.. 


WATA, MAA CDEN CIS Session 及 Cookie 的 操作 了 。 虽 然 Cookie 存 
放 于 客户 端 ， 但 是 并 不 意味 痢 Cookie 不 适合 用 于 会 员 验 证 等 场景 。 恰 恰 相 反 ， 由 于 Cookie 
存放 于 客户 端 ， 使 得 数据 能 够 长 时 间 保 存 〈 除 非 用 户主 动 柚 除 )， 这 束 意 味 看 只 要 用 户 登 录 
一 次 ， 在 很 长 一 段 时 间 内 束 不 需要 和 草 复 进行 登录 (通过 配置 COOKIE_EXPIRE 选项 实现 )， 
这 对 用 户 而 言 无 疑 是 良好 的 体验 。 所 以 很 多 网 站 会 使 用 加 密 的 方式 存放 Cookie 用 户 名 及 密 
亿 ， 甚 至 结合 软件 与 硬件 签名 实现 高 度 安 全 的 Cookie。 总 而 言 之 ， 使 用 Cookie 存放 登录 数 
据 是 可 行 的 ， 但 要 做 好 加 密 与 解密 相关 操作 。 


9.5 ”小结 


本 和 草 一 开始 首先 深入 介绍 了 当前 最 主流 的 几 短 XML 处 理 引 擎 ， 从 原理 到 实战 ， 详 细 
地 讲解 了 SAX, DOM, SimpleXML 的 方方面面 ， 帮 助 谈 者 提高 对 XML MAR. RETN 
绍 了 人 简洁 实用 的 Json 数据 ， 并 结合 Jquery 详细 介绍 了 利用 Json 进行 前 后 台 分 工 合 作 的 过 
程 ， 由 于 Jsonp 是 Json 异步 请 求 的 通信 方式 ， 所 以 本 章 还 对 Jsonp 的 原理 及 使 用 过 程 做 了 
全 面 的 介绍 。 

最 后 ， 深 入 介绍 在 MVC 开发 中 的 Session 及 Cookie 的 操作 ， 这 是 网 站 开发 中 最 名 使 用 
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的 功能 ， 也 是 重要 的 功能 ， 所 以 笔者 重点 介绍 了 Session， 并 结合 MySQL 数据 库 ， 详 细 介 绍 
了 handler 驱动 扩展 ， 并 将 Session 储存 到 数据 库 。 

本 章 涉 及 的 PHP 额外 知识 比较 多 ， 例 如 DOM, Jquery 等 。 建 议 读者 在 学 习 过 程 中 务必 
莽 成 民 好 的 记录 笔记 习惯 ， 如 果 遇 到 到 本 书 没 有 介绍 的 知识 ， 可 以 暂且 记 下 ， 并 通过 其 他 资 
料 补 齐 所 需 知 识 。 下 一 章 将 介绍 网 站 缓存 功能 ， 网 站 缓存 不 是 必需 的 ， 但 如 果 要 让 网 站 运行 
得 更 加 稳定 及 高 效 ， 使 用 缓存 功能 是 必需 的 。 
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PHP 是 一 种 脚本 语言 ， 也 是 一 种 动态 语言 ， 所 请 的 动态 语言 是 由 Web 服务 器 解释 ， 然 
后 输出 HTML， 用 户 最 终 看 到 的 是 动态 语言 运算 后 的 结果 。 运 算 的 过 程 ， 根 据 程序 逻辑 的 复 
杂 情 况 而 有 所 不 同 ， 响 应 速度 也 就 不 一 样 ， 如 果 多 个 用 户 同 时 访问 同一 页 面 ，Web 服务 器 都 
为 每 个 用 户 执 行 一 次 运算 过 程 ， 这 种 效率 是 非常 低 的 ， 在 人 气 少 的 网 站 不 会 成 为 问题 ， 但 访 
问 量 过 高 时 通常 会 导致 性 能 瓶颈 。 

数据 缓存 能 够 在 一 定 程 度 上 提高 网 站 的 运行 速度 ， 降 低 服务 器 压力 。 本 章 将 详细 介绍 
ThinkPHP 内 置 的 缓存 功能 ， 包 括 文件 缓存 、 内 存 缓存 等 。 通 过 本 章 的 学 习 ， 在 开发 大 并 发 
网 站 时 将 会 得 心 应 手 。 


学 习 目 标 


了 解 绥 存 概念 。 

了 解 ThinkPHP 缓存 中 间 件 。 

笃 握 文件 缓存 的 使 用 。 

掌握 Memcached 的 安装 与 使 用 。 

了 解 Redis 绥 存 扩展 。 

掌握 Redis 的 安装 及 简单 使 用 。 

掌握 ThinkPHP 静态 绥 存 的 配置 与 使 用 。 
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10.1 Cache 类 


Cache 类 是 ThinkPHP 内 置 的 绥 存 中 间 件 ， 提 供 了 对 外 操作 绥 存 的 所 有 方法 。Cache 类 
本 喘 不 参与 缓存 的 实际 操作 ， 所 有 的 绥 存 操作 需要 由 相应 的 绥 存 张 动 进行 。 与 Session 驱动 
一 样 ，Cache 绥 存 驱动 也 是 能 够 扩展 的 ， 但 系统 默认 已 经 内 置 了 当前 所 有 主流 的 缓存 驱动 ， 
开发 人 员 只 需要 在 配置 文件 中 ， 或 者 初始 化 Cache 类 时 指定 驱动 即 可 。 下 面 首先 介绍 Cache 
es 


10.1.1 缓存 的 方式 


在 ThinkPHP 中 ， 绥 存 方式 主要 分 为 两 种 : 一 种 基于 文件 模式 〈 默 认 模 式 ); 另 一 种 基 
于 扩展 驱动 模式 。 使 用 文件 模式 的 文件 缓存 不 需要 特殊 配置 ， 只 需要 缓存 目 录 可 谈 写 即 
可 。 前 面 提 到 过 Cache 中 间 件 只 是 一 个 对 外 接口 ， 并 不 参与 绥 存 管理 ， 绥 存 过 程 如 几 10-1 


所 不 。 
Ge 是 否 指定 驱动 名 ? 进入 驱动 模式 5 


Cache 中 间 件 


否 
使 用 文人 
IO 操 作 


图 10-1 Cache 中 间 件 处 理 绥 存 请 求 过 程 


如 图 10-1 所 示 ， 在 应 用 请 求 缓 存 操作 时 ， 首 先 由 Cache 中 间 件 统一 收集 信息 ， 如 条 系 
统 配 置 文件 中 没有 指定 绥 存 驱动 ，Cache 中 间 件 会 调用 文件 系统 处 理 缓存; 如果 配置 信息 中 
指定 有 相应 的 绥 存 驱动 ， 并 且 驱 动 类 中 包含 相应 的 驱动 实现 ， 那 么 系统 将 会 进入 驱动 模式 。 
最 终 开发 人 员 不 需要 针对 特定 的 缕 存 驱动 调用 相应 的 方法 ， 而 只 需要 统一 调用 Cache 中 间 件 
中 提供 的 公开 方法 即 可 ， 例 如 sets get 等 。 通 过 前 面 的 介绍 ， 可 以 看 到 ThinkPHP 提供 的 绥 
存 模块 是 非常 完善 鸣 ， 接 来 下 将 进一步 介绍 其 实战 用 法 。 


10.1.2 ”开启 缓存 

绥 存 不 需要 手动 开启 ， 默 认 情况 下 系统 已 经 开启 。 当 处 于 非 调 试 模式 时 ， 系 统 还 会 对 数 
据 表 字段 、 模 板 文件 等 进行 缓存 。 对 于 程序 开发 人 员 而 言 ， 只 需 在 配置 文件 中 对 缓存 参数 进 
行 配 置 即 可 ， 用 于 配置 缓存 的 参数 如 表 10-1 所 示 。 
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表 10-1 缓存 配置 参数 


绥 存 配置 ù 默 认 值 
DATA CACHE TYPE BRA file 
DATA CACHE PATH 绥 存 存放 路 径 appName/Runtime/Temp 
DATA CACHE SUBDIR 是 否 使 用 子 目 录 存 放 组 存 false 
DATA CACHE TIME 绥 存 默认 过 期 时 间 3600 
DATA CACHE COMPRESS 是 否 对 数据 进行 压缩 false 
DATA CACHE CHECK 是 否 对 缓存 进行 检验 false 


通常 情况 下 ， 只 需要 设置 DATA CACHE TIME 选项 即 可 。 在 使 用 缓存 前 ， 还 需要 对 
Cache 类 进行 初始 化 ， 如 以 下 代码 所 示 。 


public EE 
ŞCache=Cache: :getInstance (); 


} 

Cache 类 是 系统 的 基础 类 库 ， 使 用 时 不 需要 导入 。getInstance 方法 用 于 初始 化 缓存 引 
擎 ， 并 允许 传 入 初始 化 参数 优先 级 大 于 配置 文件 )， 该 方法 的 参数 格式 如 下 。 

getInstance(' 缕 存 方式 ',' 绥 存 参 数 ') 

其 中 参数 äre rd, ZE DATA CACHE TYPE 配置 项 ， 参 数 2 以 关联 数 
组 的 方式 传 入 表 10-1 配置 参数 。 

初始 化 完成 后 ， 聊 可 以 对 缓存 进行 增 、 删 、 改 、 得 了 ， 如 以 下 代码 所 示 。 


public function incex (|) < 


ŞCache=Cache: :getInstance(); 

if (Cache <et username, EI) 
Şrows=$Cache->get ("username"); 
echo $rowsSg 


} 


} 

读者 可 以 在 /Runtime/Temp 目录 得 看 到 绥 存 后 的 文件 。 对 于 Cache 中 间 件 的 操作 ， 系 
统 还 提供 了 快捷 函数 S。 使 用 快捷 函数 处 理 缓存 ， 将 有 效 地 提高 开发 效率 ， 如 以 下 代码 
Oe 


publie function incex () 4 
S('username','#F'); 


IE (Srows > (Uúusernamen)) i 


echo Srows; 


} 


j 
上 述 代 码 使 用 快捷 函数 实现 了 绥 存 号 入 与 获取 ， 最 终结 朱 与 直接 使 用 Cache 中 间 件 是 一 
样 的 。 前 面 介绍 的 都 是 默认 文件 系统 缓存 ， 接 下 来 将 结合 Memcached 内 存 数据 库 介 绍 驱动 


10.1.3 ”安装 Memcached 


Memcahced 是 一 余 小 巧 、 高 效 且 成 玖 的 内 存 数 据 库 。 与 普通 的 数据 库 不 同 ，Memcached 
存放 的 数据 只 能 是 简单 的 键 值 对 ， 在 查询 时 需要 根据 存放 的 key 获取 数据 。Memcached 最 大 
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的 特点 是 数据 存放 于 内 存 。 众 所 周知 内 存 是 计算 机 上 数据 储存 最 快 的 单元 器 件 之 一 ， 如 果 将 
数据 存放 于 内 存 中 ， 将 会 获得 比 传统 文件 系统 高 10 倍 效率 的 读 写 性 能 。 所 以 Memcached 通 
和 是 高 效 存 储 的 代名词 ， 在 很 长 一 段 时 间 内 ，Memcached 都 是 各 大 型 门户 网 站 所 采用 的 绥 存 
系统 。ThinkPHP 对 Memcached 的 文 持 已 经 非常 完善 ， 在 使 用 前 首先 需要 正确 安装 并 运行 ， 
e rd Var Linux 平台 及 Windows FS, FEADAIN. 

1. 在 Linux 平台 上 安装 Memcached 

Memcached 的 安装 比较 人 简单， 但 如 果 要 让 PHP 支持 Memcached， 还 需要 安装 
Memcache for PHP 扩展 。 当 然 ， 读 者 也 可 以 将 Memcached 独立 安装 到 一 台 服 务 费 ， 这 里 假 
设 使 用 的 是 XAMPP 开 友 环境 包 ， 现 在 第 要 安 北 Memcached， 步 又 如 下 所 示 《〈 以 Centos 6.0 
为 例 )。 

(1) 安装 libevent 

与 前 面 介绍 的 HTTPSQS 一 样 ，Memcached 也 是 使 用 异步 事件 处 理 的 ， 所 以 在 安装 
Memcached 之 前 ， 首 先 需 要 答 看 当前 环境 是 合 已 经 安 闭 异步 事件 处 理 冰 数 库 libevent。 

[root@~]# ls -al /usr/local/lib | grep libevent 


libevent-1.4.s0.2 -> libevent-1.4.so.2.1.3 
如 果 没 有 ， 可 以 使 用 yum 进行 安装 ， 但 版 本 可 能 比较 低 。 这 里 将 使 用 源 代 人 码 安 装 ， 安 
北 步 缀 如 以 下 命令 所 示 。 


[root@~]# mkdir /opt/data 

root@~]# cd /opt/data 

root@~]# wget http://soft.beauty-soft.net/lib/libevent-1.4.13-stable.tar.gz 
root@~]# tar -zxvf libevent-1.4.13-stable.tar.gz 

root@~]# cd libevent-1.4.13-stable 

root@~]#./configure 

root@~]# make 

[root@~]# make install 


默认 会 被 安装 到 /usr/local/libevent-2.0.12-stable/ 目 录 下 。 

(2) 安装 Memcached 

接 下 来 就 可 以 安装 Memcached 主 程序 了 。Memcached 官方 网 站 为 http://memcached.org/， 
读者 可 以 在 该 网 站 上 找到 最 狐 的 源 代 人 码 安 冯 包 ， 这 里 将 使 用 1.4.15 版 本 ， 安 北 过 程 如 下 。 
[root@~]# cd ../ 
[root@~]# wget http://soft.beauty-soft.net/lib/memcached-1.4.15.tar.gz 
[root@~]# tar -zxvf memcached-1.4.15.tar.gz 
[root@~]#./configure 


make 


# 
# 
# 
[root@~]# cd memcached-1.4.15 
# 
[root@~]# 
# 


[root@~]# make install 
默认 情况 下 ，Memcached 将 被 安装 到 /usr/local/bin/ 目 录 下 。 在 启动 前 ， 还 需要 将 libevent 
包 存 放 路 径 加 入 到 /etc/ls.so.conf 文件 中 ， 如 以 下 命令 所 示 。 


[root@~]# vi /etc/ld.so.conf 
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include ld.so.conf.d/*.conf 
/usr/local/lib 
完成 上 述 操作 后 ， 现 在 就 可 以 局 动 Memcached 主 程序 了 ， 如 以 下 命令 所 示 。 
/usr/local/bin/memcached -m 100 -p 11211 -d -u root -P /tmp/memcached.pid -c256 
在 局 动 时 ， 第 用 的 司 动 参数 如 下 。 
> -m: 最 大 使 用 内 存 ， 以 MB 为 单位 ， 默 认 64。 
> -p: Memcached 局 动 进程 所 使 用 的 TCP 通信 交口 ， 默 认 11211. 
d: 将 Memcached 作为 后 合 守护 进程 运行 。 
: Hahn, 
-P: 进程 文件 存放 路 径 。 
-c: 最 大 运行 并 友 数 ， 默 认 1024。 
> -1l: 监听 服务 喜 地 址 〈 即 允许 tente 登录 的 IP). 
启动 完成 后 ， 可 以 直接 查看 主 进程 是 否 存 在 ， 判 断 是 否 局 动 成 功 。 


[root@~]# pstree |grep mem 


VW VY 
= 


| -memcached---5* [ {memcached} ] 
(3) 安装 Memcache 扩展 
Memcache 是 PHP rä, EKRA m ne CAHE PHP 环境 。 
为 只 有 将 Memcache 编译 到 PHP 扩展 模块 中 ， 开 发 人 员 才 能 使 用 PHP 代码 调用 Memcache, 
PHP 内 置 了 一 个 编 详 第 三 方 扩展 的 工具 phpzie， 接 下 来 束 利 用 该 工具 来 编 详 Memcache， 步 
又 如 下 。 
首先 下 载 和 解压 Memcache， 这 里 使 用 的 版 本 是 2.2.5， 命 令 如 下 所 示 。 


[root@~]# wget http://soft.beauty-soft.net/lib/memcache-2.2.5.tgz 


[root@~]# tar -zxvf memcache-2.2.5 


[root@~]# cd memcache-2.2.5 

接 下 来 使 用 phpzie 工具 配置 Memcache。 假 设 PHP 安装 路 径 为 /ust/local/php/， 那 么 
phpize 路 人 径 束 是 /usr/local/php/bin/phpize。 
[rootte-]} 7usr/local/php/bin/phpize 


[root@~]#./configure --with-php-config=/usr/local/php/bin/php-config 

现在 就 可 以 使 用 make 工具 编译 Memcache 了 ， 过 程 如 下 。 

[root@~]# make 

[root@~]# make install 

Installing shared extensions: /usr/local/php/lib/php/extensions/no-debug-non- 

zts-20090626/ 

扩展 将 被 安装 到 /usrvlocalphp/lib/php/extensions/no-debug-non-zts-20090626/ 目 孙 。 最 后 还 
需要 修改 php.ini 配置 项 ， 将 该 扩展 目录 加 入 到 extension dir WW. EKE php.ini 已经 默认 
加 入 了 该 扩展 目录 ， 只 是 被 注释 而 已 ， 只 需要 去 挥 注释 即 可 ， 过 程 如 下 。 


[root@~]# vi /usr/local/php/etc/php.ini 


805 ; Directory in which the loadable extensions (modules) reside. 


806 ; http://php.net/extension-dir 
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807 extension dir = "/usr/local/php/lib/php/extensions/no-debug-non-zts- 
20090626/" 

808 extension = "memcache.so" 

809 extension = "pdo mysql.so" 


读者 也 可 以 手动 添加 上 。 保 存 php.ini 配置 文件 ， 重 启 php-fpm 或 者 web 服务 器 ， 以 便 
配置 文件 生效 。 

[root@~]# /etc/init.d/php-fpm restart 

至 此 ，Memcached 及 Memcache 都 已 经 安装 完成 ， 通 过 phpinfo 函数 输出 信息 将 可 以 看 
到 Memcache 配置 项 ， 如 图 10-2 ra 


eive persistent eonnections ER 


图 10-2 WIT Memcache 扩展 


2. 在 Windows 平台 上 安装 Memcached 

在 Windows 下 安装 Memcached 主 程序 及 Memcache 扩展 比较 简单 ， 使 用 方式 不 论 在 任 
何平 台 上 都 没有 区 别 。 但 是 无 论 是 官方 文档 还 是 第 三 方 权威 资料 ， 都 不 建议 在 Windows 下 
ZIL Memcached 生产 环境 。 通 常情 况 下 ， 在 Windows 上 只 用 于 开发 环境 。 下 面 将 简单 介绍 
Æ Windows 7 下 的 安装 过 程 。 

自 先 需 要 下 载 Windows 安装 包 。 

http://beauty-soft.net/book/php mvc/down/memcached-windows.html 

将 压缩 包 解 压 到 指定 的 目录 ， 例 如 Ci:memcached， 进 入 该 目录 将 会 看 到 memcached.exe 
主 程序 。 接 下 来 使 用 命令 终端 执行 memcached.exe， 命 令 如 下 。 

c:\memcached> memcached.exe -d install 

通过 上 述 步骤 ，Memcached 就 目 动 安 状 完 成 了 ， 接 来 下 只 需要 启动 即 可 。 

c:\memcached> memcached.exe -d start 

启动 完成 后 ， 可 以 通过 任务 管理 器 查看 到 Memcached 进程 ， 也 可 使 用 netstat -an 查看 
11211 get, al Memcached 是 否 安 北 成 功 。 接 下 来 只 需要 安装 PHP 扩展 即 可 。 

首先 下 载 memcache.dll 扩展 ， 这 里 使 用 的 版 本 为 2.2.5. 

http://soft.beauty-soft.net/lib/win/php memcache-2.2.6-5.3-vc9-x86.zip 

解压 后 将 得 到 php_memcache.dll 扩展 文件 。 将 php memeache d 复制 到 PHP 扩展 目 
录 ， 通 党 该 目录 位 于 PHP 安装 目录 下 的 er Ha CEH php\ext)。 打 开 php.ini 配置 文件 ， 将 
php_memcache.dll 扩展 加 入 到 配置 项 中 ， 如 以 下 代码 所 示 。 


pextensionsphp Ciey eli 


extension=php miac.oll 
extension=php_memcache.dl1 


保存 配置 文件 ， 重 局 Web 服务 右 ， 通 过 phpinfo 函数 可 以 看 到 Memcache 选项 ， 证 明 
PHP J EZR. 
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10.1.4 使 用 Memcached 


maA, ThinkPHP ARCA Memcache RFI. FRA RNR mE 
指定 DATA CACHE TYPE WEM, Win UKER. HuK DATA CACHE TYPE 值 
配置 为 memeache, MARRS 0 dë re OR DI Memcache, 

为 了 帮助 读者 提高 对 Memcached 绥 存 张 动 的 认识 ， 这 里 将 首先 使 用 传统 的 PHP ICH 
JE Memcached; 然后 再 介绍 使 用 telnet 命令 管理 Memcached; 最 后 将 介绍 使 用 MVC 操作 
Memcached。 

1. PHP 操作 Memcached 

成 功 安 装 Memcache 扩展 后 ，PHP WHE Memcached 通信 的 能 力 了 。 首 先 需 要 使 用 
content 方法 连接 上 Memcached 服务 器 ， 人 代码 如 下 所 示 。 


<?php 
Smem = new Memcache; 
Smem=>connecTt (v1i27.0.0.,.17, 11211)7 


ÉTRE B A ER ESTE, ERRE WM. o. E eR A EH 
Memcache DN Go AVAX) REITA EEIT BEE - 

(1) add 增加 绥 存 条 目 ) 

add 方法 可 以 问 Memcached 服务 器 增加 绥 存 条 目 ， 如 末 存 在 相同 缓存 条 目 ， 则 放弃 操 
作 。 代 人 码 如 下 所 示 。 


<?php 
header (" Contcent=Type: text/html; Charset=0EE-8") 
Smem = new Memcache; 
Smem=->connecTt (127.0.0.17, 11211)s 
smem->add ("kevy"，" 现 在 时 间 " .time()，0，60) ; 


d 

参数 1 为 缓存 key, EK 250 个 碳 文 字符 之 内 ， 访 参数 需要 唯一 性 ;参数 2 表示 绥 存 内 
Z WA 1000KB， 接 受 钟 见 的 数据 类 型 ， 包 括 数组 、Json 等 ;参数 3 表示 是 否 使 用 zlib 压 
纺 数据 ， 默 认为 不 使 用 ; 参数 4 表示 缓存 过 期 时 间 ， 以 秒 为 单位 ， 最 大 值 为 2592000 CER 
30d). 

(2) delete HIIRI ERT) 

delete 可 以 根据 传 入 的 key， 删 除 指定 的 缓存 条 目 。 


<?php 
header (" Content-Type: text/html; charset=utT=6" ) 
Smem = new Memcache; 
Smemn—>Sconneect (*127-0-0-1"; 11211) 7 
smem->delete ("key") 


(3) set (修改 ) 
set 方法 与 add 方法 类 似 ，set 方法 用 于 修改 或 添加 绥 存 条 目 。 如 果 数 据 库 中 没有 对 应 的 
key， 则 将 该 条 数据 存放 到 数据 库 中 ， 否 则 对 指定 的 数据 进行 修改 。 


<?php 
header (" Contcent=Type: text/html; charsetc=utt=8" ) 


Smem = new Memcache; 
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Smem=> Connect EE EES 
Smem-—Set(mkeun, THER Cimeii, 0, c0); 
2> 


(4) get (查询 ) 
get 方法 可 以 对 指定 的 key 取得 对 应 的 value. 


<?php 
header (" Contcent=Type: text/html; charsert=uti=6" ) s 
Smem = new Memcache; 
Smem=> Connect (127-00-01; 11211)7 
smem->get ("key", 0); 
E 


get 方法 文 持 两 个 参数 ， 其 中 参数 1 表示 缓存 key; 参数 2 用 于 解压 缩 或 反 序 列 ， 人 参数 2 
共有 3 个 选项 ， 如 下 所 示 。 

> 0: 数据 没有 经 过 压缩 〈 默 认 )。 

> 1: 数据 已 经 经 过 序列 化 ， 需 要 反 序 列 数据 。 

> 2: 数据 已 被 zlib 压缩 ， 需 要 解压 缩 。 

2. 使 用 命令 行 管理 Memcached 

直接 登录 Memcached 服务 右 ， 能 够 直观 地 奋 看 到 数据 库 储 存 状 态 ， 对 数据 进行 管理 
等 。 在 程序 调试 阶段 是 必 不 可 少 的 操作 之 一 ， 所 以 谈 者 需要 掌握 一 些 频 老 使 用 的 命令 ， 方 便 
在 程序 开发 中 进行 调试 。Memcached 有 第 三 方 完善 的 管理 工具 ， 但 最 方便 的 还 是 使 用 
telnte， 命 令 如 下 。 

C:>telnet localhost 11211 

成 功 登 录 后 残 可 以 对 数据 库 进 行 管理 了 ， 第 用 的 命令 有 stats, stats reset, stats slavs, 
stats items, stats cachedump、set、get、gets 等 ， 命 令 说 明 如 下 。 
stats: 获取 Memcached 服务 堪 状态 信息 ， 包 括 服务 器 绥 存 数量 ， 内 在 占用 等 。 
stats reset: 重新 统计 数据 。 
stats slabs: 显示 slabs» 
stats items: 显示 当前 slabs 中 存放 的 所 有 绥 存 项 。 
stats cachedump: 显示 slabs 指定 范围 内 的 缓存 项 ,例如 stats cachedump 0,3. 
flush all: 清空 当前 服务 占 上 所 有 绥 存 数据 。 
quit: 退出 当前 连接 。 
set: Books In e Zë re In. 
get: 根据 key 返回 绥 存 项 内 容 。 
在 MVC 中 操作 Memcached 

在 ThinkPHP 中 使 用 Memcached 是 非 冲 简单 及 高 效 的 ， 系 统 默认 已 经 提供 了 Memcache 
绥 存 驱动 ， 要 将 缓存 模式 改 为 Memcache， 只 需要 修改 配置 文件 即 可 ， 如 以 下 代码 所 示 。 


'DATA CACHE _ TYPE'=>'Memcache', // 组 存 驱动 


Vv 


VW WWW WW V 


9 


CE 
ET // 服 务 器 端口 
DATA CACHE TIMEOUT! 26000. // 过 期 时 间 


在 切换 到 Memcache 驱动 后 ， 原 有 的 开发 代码 不 需要 做 任何 改变 ， 这 也 是 Cache 中 间 件 
最 为 灵活 和 方便 的 地 方 。 
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10.2 Memcached 实战 应 用 


绥 存 虽然 能 够 提高 页 面 的 啊 应 速度 ， 但 并 非 所 有 场合 都 适合 使 用 缓存 。Memcached 是 一 
个 内 存 绥 存 数据 库 ， 对 数据 的 容量 及 长 度 都 有 一 定 的 限制 ， 所 以 Memcached 并 不 适合 组 存 
单项 数据 量 过 大 的 数据 。 通 常情 况 下 ，Memcached 适合 于 绥 存 SQL 语句 、 数 据 集 、 用 户 临 
时 性 数据 、 延 人 运 查 询 数 据 以 及 Session 等 。 也 就 是 说 ， 绥 存 只 适用 于 查询 操作 ， 并 不 适合 添 
加 、 人 和 修改、 删除 ， 这 一 点 需要 注意 。 下 面 将 通过 几 个 典型 的 示例 ， 介 绍 Memcached 的 实战 
应 用 ， 加 深 对 数据 绥 存 的 认识 。 


10.2.1 页 面 局 部 缓存 


有 些 动态 页 和 面 ， 残 算 使 用 了 静态 化 处 理 〈 例 如 生成 HIML)7， 也 不 能 很 好 地 解决 局 部 载 
入 过 慢 的 问题 ， 这 时 束 可 以 考虑 使 用 页 和 面 局 部 绥 存 。 页 面 局 部 绥 存 可 以 在 一 定 程度 上 减 小 服 
务 咒 的 压力 ， 提 高 页 面 整体 啊 应 速度 。 

所 以 现在 主流 的 Web 服务 右 〈 例 如 Nginx) 或 者 绥 存 服务 器 〈 例 如 Varnish, Squid) 都 
提供 了 页 面 局 部 绥 存 的 功能 ， 但 是 使 用 这 些 技术 需要 额外 的 便 件 成 本 ， 这 些 技术 都 是 基于 服 
Far IO 实现 的 ， 在 性 能 上 不 及 Memcached， 所 以 开发 人 员 也 可 以 使 用 Memcache 实现 少量 
Il ed 

由 于 Memcache 对 数据 的 限制 ， 押 以 只 需要 绥 存 页 面 中 得 询 率 过 高 但 更 新 较 少 的 区 域 即 
可 ， 例 如 Ajax、 栏 目 导 航 信息 、 消 息 提 示 等 。 如 以 下 代 但 所 示 。 


publie function index () q 


LE (1S("navigation™)) 
Sml = 下 
EE parser new can miClass 


mpa rsen Parsel mh, 
onov oml parser- data; 
// 缓 存 xml 部 分 数据 
S(navigation", Snav); 

}elsel 
Snav=S ("navigation"); 

} 

Sthis=>dLsplay ("nav , Snav)s 

Schis >cisoleay O; 

li 


10.2.2 ”缓存 数据 集 


在 ThinkPHP 中 ， 绥 存 数据 集 是 最 常用 及 最 有 意义 的 一 项 操作 ， 因 为 系统 内 置 了 非常 智 
能 化 编译 及 绥 存 机 制 ， 数 据 表 字段 、SQL 语句 等 在 非 调试 模式 下 会 目 动 进行 缓存 ， 所 以 开发 
人 员 不 需要 手动 处 理 。 绥 存 数据 集 能 够 避免 相同 数据 多 次 该 取 数 据 库 ， 从 而 提高 查询 效率 。 

通常 情况 下 ， 数 据 集 是 由 多 条 记录 组 成 的 ， 所 以 只 需要 将 一 条 记录 作为 一 条 Memcached 
数据 项 ， 就 可 以 有 效 改善 查询 速度 ; 如 果 数 据 量 比 较 小 ， 可 以 一 次 性 将 整个 数据 集 进 行 绥 
存 。 绥 存 数据 集 ， 可 分 为 3 种 模式 ， 下 面 分 别 介 绍 。 
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1. 缓存 数据 集 字段 

缓存 数据 集 字 段 是 效率 最 高 、 也 是 Memcached 最 常 使 用 的 一 种 缓存 模式 。 绥 存 数据 
集 ， 只 需要 在 第 一 次 返回 结果 时 ， 将 其 中 没有 创建 索引 ， 或 者 数据 量 比较 大 的 字段 〈 例 如 
text, varchar 字段 等 ) 缓存 到 Memcached 中 ， 用 户 再 次 查询 相同 的 数据 时 ， 系 统 将 直接 输出 
已 经 缓存 过 的 字段 值 ， 避 人 免 再 次 查询 数据 库 。 如 以 下 代码 所 示 。 


public function index (){ 
ŞarticleO0bj=M ("Article"); 
Siele=array (icv, "Eitle", "content ™, ) g 
Srows=$article0oj=>field (Siriel) =>selecrt 人) ; 
foreach (Srows as Skey=>svalue) { 
Stictle=Ṣ$yalue[“title™"] 7 
Scontent=$yaLlue |" content™] 
ii 
// 绥 存单 条 标题 
ee 
Lelzel 
/ /获取 标题 绥 存 
Eege 
i 
me ne el el el 
// 缓 存单 条 内 容 
ERR 
}elselfl 
/ /获取 内 容 绥 存 


人 


} 
EE Ee 
Srowlzkevl "content" | =$contents; 
l 
Şthis=>assign("LLsSt", $row) 5 
Sthis >display (Q); 


} 

上 述 代 码 中 ， 将 title 及 content 字段 值 使 用 Memcached 绥 存 为 一 个 独立 项 ， 再 次 查询 
title 及 content 字段 时 ， 系 统 将 不 会 从 tpk article 数据 表 获 取 ， 而 是 直接 在 绥 存 系统 中 获取 ， 
很 好 地 提高 了 效率 。 

不 仅 如 此 ， 由 于 绥 存 系统 是 全 局 性 的 ， 这 也 驳 意 味 看 绥 存 系统 中 的 数据 可 以 在 网 站 内 任 
何 地 方 调用 。 根 据 这 个 原理 ， 在 单 击 文章 详情 时 ， 只 需要 输出 Memcachd 中 的 缓存 即 可 ， 而 
不 需要 查询 数据 表 。 如 以 下 代码 所 示 。 


/** 
* 文章 详情 页 面 
SCH 
Buaolie funetieon cetaileci() i 
EE el 
”owsl" title= ("article title ke 
Eeer eegene EE KE 
Jelse{ 


ŞarticleO0bj=M ("Article"); 
Sfiele=array (vi title", "content, E 


0 0 
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Sthis- >assion ("content $rows)s 
Sthis=>adLsplay() s 


} 

如 上 述 代 但 所 示 ， 在 文章 详细 页 面 中 ， 根 据 GET 传递 参数 判断 绥 存 系统 是 否 存 在 相应 
的 数据 ， 如 有 果 存 在 ， 则 和 直接 返回 数据 ， 不 再 操作 数据 库 。 经 过 上 述 处 理 ， 数 据 碍 询 速度 将 得 
到 质 的 提升 。 

2. 缓存 数组 集结 果 

如 果 数 据 集 返回 的 数据 量 比较 小 ， 可 以 将 全 部 结果 绥 存 到 Memcached 中 ， 这 样 做 的 好 
处 是 可 以 提高 开发 效率 ， 同 时 也 能 够 提高 得 询 速度 。 但 也 有 缺点 ， 如 条 返回 的 数据 量 突然 增 
多 ， 其 至 超过 Memcache 的 容量 ， 那 么 数据 集 将 会 丢失 数据 ， 造 成 系统 运行 异 稼 。 

同时 需要 注意 ， 绥 存 整 个 数据 集 必须 要 将 结果 进行 序列 化 (要 确保 获取 时 能 够 完美 反 序 
列 化 )， 币 用 的 序列 化 函数 有 json encode、sizeo、base64 encode 等 。 绥 存 数 据 集 相 对 比较 简 
单 ， 只 需要 创建 缓存 Key 即 可 ， 如 以 下 代码 所 示 。 


public funeeion incex EE 
Sartcicleoboj=M(WArticle") y 
impor e Re ER 


ou aree lo ee arra adier eerca com 

SPage = new Page ($count,25); 

SE RE e el ee Ee ee s 

ee Eer Ger better ee geg eet ele 
EE Shi 

if (empty ($thisPage) || SthisPage==0 ) S$thisPage=1; 


JI a e 
// 根 据 当 前 分 页 编号 创建 缓存 集 


DEE e aA ene de 


Eee 
pelse{ 
// 直 接 谈 取 缓存 
-rows Joon EE E 


SEI GE Met eebe e 
Şthis->display(); 


} 

上 述 代 三 使 用 到 了 分 页 扩展 ， 这 里 只 需要 理解 即 可 ， 后 面 将 会 有 详细 介绍 。 一 个 数据 集 
束 相 当 于 一 个 页 面 中 的 数据 循环 体 ， 根 据 分 页 写 唯 一 性 的 特点 ， 可 以 直接 使 用 分 页 写作 为 
Memcached 绥 存 主键 ， 在 获取 数据 时 ， 一 条 Memcached 数据 项 就 是 一 个 数据 集 。 

3. 缓存 整个 数据 表 

绥 存 整个 数据 表 是 减轻 数据 库 奏 询 压 力 最 有 效 的 方法 ， 但 同时 也 是 最 消耗 内 存 的 一 种 绥 
存 方式 。 同 时 ， 由 于 在 绥 存 时 需要 全 部 该 取 数 据 表 中 的 数据 ， 然 后 执行 绥 存 操作 。 在 这 一 过 
程 中 ， 将 会 占用 服务 器 的 大 量 资 源 〈( 主 要 是 内 存 及 CPU )。 

将 数据 全 部 绥 存 到 Memcached 之 后 ， 程 序 在 获取 数据 时 并 不 直接 连接 数据 库 ， 而 是 连 
接 绥 存 服 务 右 。 在 但 询 时 ， 只 能 根据 缓存 主键 (例如 D) 奋 询 数据 ， 并 且 只 文 持 一 个 字 
段 。 这 样 一 来 数据 库 操作 将 失去 原 有 的 灵活 性 ， 所 以 在 此 并 不 建议 使 用 这 种 模式 绥 存 数据 。 
但 如 果 确 实 需要 ， 建 议 使 用 以 下 几 种 手段 解决 上 述 提 到 的 问题 。 

> 创建 缓存 时 ， 需 要 在 访问 量 较 少 的 时 间 段 内 进行 ， 可 以 配合 计划 任务 实现 。 但 更 通 
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用 的 办 法 是 在 网 站 后 人 台中 ， 由 管理 员 手 动 生成 缓存 。 
> 由 于 Memcached 是 一 个 内 存 数据 库 ， 所 以 Memcached 所 在 的 服务 孝 内 存 必 须要 足够 
大 。 服 务 器 关机 后 ， 内 存 中 的 数据 也 将 会 被 请 衬 ， 所 以 尽量 搭建 可 容 灾 的 Memcached 
服务 器 集群 。 
> 充分 使 用 Memcached 内 置 的 zlib 压缩 功能 。 
下 面 通 过 一 段 简 单 的 示例 代码 ， 潘 示 绥 存 数 据 表 数据 的 过 程 ， 代 码 并 不 完善 ， 在 真实 应 
用 开发 中 还 需要 读者 继续 完善 ， 代 码 如 下 所 示 。 
/** 
* 生成 缓存 
* Enter description here ... 
publie funcecion acd!) cache () i 


ŞarticleO0bj=M ("Article"); 
Şrows=Ş$article0bj->select(); 


foreach ($rows as Şkey=>Ş$value) { 
// 逐 条 添加 
ee 
Lee ASS EE 
"rtleit-zzGualuefiritrlent, 
content zz äualuelicontentnl, 
ecd meet va es 
dem = valle e eae 
II: 
} 


} 
生成 缓存 后 ， 只 需要 根据 cache id 进行 获取 即 可 ， 如 以 下 代码 所 示 。 
/** 
* 获取 缓存 
x Enter description here ... 
SE 
publie function get cache () i 
EIERE enpe EE 
EE Els 
for (ëieëmin ld el- mas Ido) 
CaCO Eey aa e le; 
ea ele 
} 
ŞSthis=>assign ("list $data) s 
ŞStcnis=>display() 7 


} 
上 述 代 人 码 使 用 for 区 间 批 量 获取 ， 读 者 可 以 根据 前 面 介 绍 的 第 2 种 绥 存 模式 获取 单条 组 
存 数 据 。 
10.2.3 ”使 用 Memcache 存放 Session 


表面 介绍 的 都 是 数据 库 的 缓存 ， 接 下 来 将 介绍 使 用 Memcached 绥 存 Session。 无 论 是 使 
用 哪 种 技术 或 者 开发 框架 ，Session 的 储存 都 是 使 用 文件 操作 来 完成 的 。 
PHP 默认 将 Session 存放 于 系统 临时 目录 下 【例如 /tmp)， 如 果 会 话 大 量 产 生 ， 这 无 颖 对 
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服务 占 的 IO 性 能 是 一 个 苛刻 的 挑战 。 现 在 可 以 利用 Memcached 的 高 性 能 ， 将 大 量 的 Session 
绥 存 到 绥 存 服务 占 中 ， 由 于 在 内 存 中 完成 所 有 会 话 操 作 ， 性 能 将 得 到 大 幅 提 升 。 

此 外 ， 大 型 网 站 通 钊 都 是 使 用 负载 均衡 来 文 撑 运 管 的 ， 各 服务 器 的 文件 系统 是 独立 的 ， 
IMARA- EHHA, RAK Session 将 失效 ， 这 无 疑 是 糟糕 的 用 户 体 验 。 传 统 PHP 开 
发 可 以 使 用 文件 共享 、Session 存 数据 库 以 及 Cookie 存 Session 解决 。 但 无 论 从 性 能 还 是 安全 
性 考虑 ， 基 于 TCP 协议 的 Memcached 部 具有 优势 ， 唯 一 的 缺点 束 古 需要 你 证 Memcached 服 
务 占 的 高 可 菲 性 运行 。 

上 一 章 已 经 介绍 过 PHP 内 置 的 session set save handler 疯 数 ， 该 函数 用 于 扩展 内 置 的 
Session 操作 ， 例 如 读 取 、 写 入 、 修 改 等 ， 接 下 来 将 利用 session set_save_handler 函数 ， 实 现 
将 Session 存放 于 Memcached。 

首先 在 ThinkPHP/Extend/Driver/Session 目录 下 创建 Session 驱动 ， 并 命名 为 SessionMemcached. 
class.php， 人 代码 如 下 所 示 。 


<?php 
class SessionMemcached { 
TA 
protected) $lireTime="" ; 
// 前 组 
protected) Sprefi="" ; 
/ /Memcached DIR 
protected shander; 


x Qparam string SsavePath 
x Qparam mixed Ş$sessName 


a 
public function open ($savePath, ŞsessName) { 
Şthis->lifeTime = C" SESSION EXPRE ) $ 
Sthis=>prefi = CT Eo TONT ERER T 
Smemcache=new Memcache; 
memeacemne= connect C (TMEMCACHE EE 
if (Smemcache) { 
Şthis->hander=$memcache; 


EE Crue; 


}elsef 
return false; 
/** 
十 PE d d e e SE ES PE EE IE AAA LE EE 
* 关闭 Session 
十 TD LA die ee d e E Se Sg SE EE ES 


aaGGSsSS public 
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E 
public function Glose() i 
ee 
return Sthis->nhnandqer->close () ; 


public function reac ($sessID) 4 
$data=$this->hander->get (Ş$this->prefi.ŞsessID); 
return $data; 


* Qparam string $sessID 


* Qparam String S$sessData 


a 
public function write ($sessID,Ş$sessData) { 
Şexpire = E Ebene 


return $rthis=>hander=->se6et (SEhis >orefi., SeessID, $SsessData, expire); 


SÉ 
publie function destroy SSESSID) 4 
return Sthis=>hander=>delete (Sthis=>prefi., SessLD)s 
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x Qparam string SsessMaxLifeTime 


SCH 
public function gc($sessMaxLifeTime) { 


recurcn crues 


x Qparam string SsavePath 
x Qparam mixed $sessName 


E 
Buolie funCtion execucte() | 
EE emma (er EEN 
array (tS Enis, Close”) , 
array (&$this,"read"), 
&şthis,"write"), 
tSEhis, destroy") E 
array (e$this, EE 


array 


array 


( 
( 
( 
( 


} 
2> 


如 上 述 代 码 所 示 ， 驱 动 的 类 名 必须 与 驱动 文件 名 称 相同 ，ThinkPHP 提供 的 初始 化 接口 
方法 为 execute 方法 ， 需 要 在 初始 化 方法 中 完成 session set Save handler 函数 的 定义 。 如 果 
使 用 传统 的 PHP， 则 需要 在 目 定 义 初 始 函数 或 构造 函数 中 完成 。 

接 看 定义 session set save_handler 函数 所 需要 的 6 个 参数 ， 最 后 在 当前 类 中 分 别 实 现 参 
数 中 定义 的 6 个 方法 。 在 目 定 义 张 动 中 ， 可 以 使 用 MVC 框架 内 所 有 对 外 公开 的 类 方法 及 函 
数 。 保 存 SessionMemcached.class.php 驱动 文件 ， 将 Session 驱动 名 改 为 SessionMemcached， 


如 以 下 代 人 码 所 示 。 
CESC TON AUTO TART! true, Maz session start ee 
TO ESC TON PRERIX "Ck //Session 前 级 
'VAR SESSTON ID=- time(), //Session ID 


'SHSSTON TABLE- 'think session’, / /存放 session 的 数据 表 名 


SESS TON EXPIRE! 30000, // session 过 期 时 间 
SESSION TYPE'=>'Memcached', // 驱 动 器 名 称 


"MEMCACHE HOST=>" 1270.01"; 
"MEMCACHE PORT“ =>" 11211"; 
DATA CACHE TIMEOUT =>30000, 


由 于 在 驱动 文件 中 实现 了 session set Save_ handler 函数 所 有 参数 ， 所 以 在 使 用 时 可 以 直 
接 使 用 PHP 内 置 的 $ SESSION 获取 或 设置 Session (需要 加 前 级 ); 当然 也 可 以 使 用 
ThinkPHP 内 置 的 Session 函数 (不 需要 加 Session AJZ), WLA FRIET. 


publice cunction lacex()l 


ses on name e E 
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SE function test()f{ 
dump- TON 
} 
通过 表面 的 设置 ， 现 在 所 有 Session HE EAI Ditz. E, ARE) 都 在 
Memcached 中 完成 。 使 用 Memcached 缓存 Session， 不 仅 可 以 提高 系统 性 能 ， 由 于 
Memcached 数据 库 是 受 内 存 限 制 的 ， 一 旦 数据 达到 临界 ， 服 务 器 将 根据 时 间 顺 序 删 除 内 存 中 
的 数据 ， 在 一 定 程度 上 减少 CC 等 攻击 。 


10.3 使 用 Redis 缓存 


Redis 是 一 个 功能 强大 、 人 性 能 高 效 的 开源 数据 结构 服务 右 ，Redis 最 典型 的 应 用 是 
NoSQL。 但 事实 上 Redis 除了 作为 NoSQL 数据 库 使 用 之 外 ， 还 能 广泛 用 于 消息 队列 、 数 气 
堆栈 以 及 数据 缓存 等 众多 场合 。Redis 与 Memcached 相 类 似 ， 都 是 以 键 值 对 〈key-value) Ze 
放 数 据 的 ， 但 是 Redis 文 持 的 数据 类 型 及 特性 远 比 Memcached Za, 

在 绥 存 应 用 方面 ，Redis 同样 也 是 一 个 内 存 数 据 库 ， 拥 有 Memcached 的 快速 、 稳 定 等 特 
性 ， 并 且 文 持 数据 快照 功能 ， 开 发 人 员 可 以 通过 配置 文件 指定 数据 快照 间隔 时 间 ，Redis 会 
将 数据 快照 自动 存放 于 便 盘 中 ， 这 样 融 算 服务 器 突然 停止 服务 ，Redis 也 极 少 会 出 现 丢 失 数 
据 的 现象 。 所 以 近 些 年 越 来 越 多 的 大 型 互联 网 公司 开始 使 用 Redis 作为 缓存 服务 器 。 

ThinkPHP 内 置 了 Redis 缓存 驱动 ， 要 使 用 Redis 作为 缓存 服务 器 ， 首 先 需要 正确 安装 
Redis 数据 库 ， 然 后 安装 PHP 扩展 模块 ， 下 面 分 别 介 绍 。 


10.3.1 Redis 的 安装 


Redis 的 安装 与 Memcached 一 样 主要 分 为 两 种 安装 环境 ， 即 Windows 及 Linux。 与 多 数 
开源 软件 一 样 ，Windows 版 的 Redis 无 论 在 性 能 还 是 资源 分 配 、 线 程 稳定 性 上 都 不 及 Linux 
版 本 。 所 以 在 Windows 上 的 Redis 只 作为 程序 开发 调试 之 用 ， 并 不 建议 用 于 生产 环境 。 下 面 
将 分 别 介 绍 。 

1. 在 Windows 上 安装 Redis 

在 Windows RAEI Redis 相对 比较 简单 。 整 个 过 程 可 分 为 下 载 Redis 可 执行 文 
件 ， 创 建 或 编辑 redis.conf 配置 文件 ， 最 后 司 动 redis。 下 面 自 完 下 载 Redis. 

Windows 有 版 的 Redis 可 以 在 http://code.google.com/p/servicestack/wiki/RedisWindows 
Download 网 站 上 获取 ， 这 里 下 载 的 版 本 为 redis-2.0.0-rc4。 下 载 后 ， 打 开 redis-2.0.0-rc4.zip 
EKA, SAFAR E o 


redis-2.0.0-rc4/ 


Cygwinl.dll 
redis-benchmark.exe 
redis-check-aof.exe 


redis-check-dump.exe 


redis-cli.exe 
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redis-server.exe 

其 中 redis-server.exe 为 主 程序 ，redis-cli.exe 为 Redis 内 置 的 一 个 基于 命令 行 的 管理 工 
具 。 在 局 动 Redis 服务 进程 时 ， 需 要 指定 配置 文件 。 默 认 情 况 下 ， 压 缩 包 内 并 没有 提供 
Redis 配置 文件 ， 所 以 需要 开 友 人 员 手 动 创建 配置 文件 ， 并 命名 redis.conf， 人 代码 如 下 所 示 。 


pickile /var/run/redis. pie 
# ot 
BOrE 637/9 


# 绑 定 ip 

€ bind 127.0.0.1 

# 数据 有 效 时 间 
ESGUE s00 

# 数 据 库 数 量 
databases 16 
20 ARME ut AÄ 
save 900 1 

save 300 10 

save 60 10000 

Hae MATA EIT H i 
rdbcompression yes 
# 数据 库 文件 名 
dbfilename dump .rdb 
# Redis 工具 目录 
E 


这 里 只 是 简单 地 列 出 了 与 缓存 有 关 的 配置 项 ， 更 多 设置 可 参考 本 书 第 16 章 16.1.2 市 。 
保存 redis.conf 文件 内 容 ， 并 复制 到 redis-2.0.0-rc4 目录 ， 接 下 来 就 可 以 月 动 Redis 了 。 首 先 
在 命令 终端 中 进入 redis-2.0.0-rc4 目录 ， 例 如 cd c:redis-2.0.0-rc4， 然 后 运行 redis.serve.exe 主 
程序 ， 并 指定 配置 文件 。 

c:\redis-2.0.0-rc4>redis-server.exe redis.conf 


命令 成 功 执 行 后 ，Redis 将 以 命令 终端 的 形式 局 动 ， 如 图 10-3 ra 


C: \Users\Administrator>cd Ch 
| c:\>cd redis-2.0.0-rc4 


c:\redis-2.0.0-rc4>redis-server .exe redis.conf 

Dr Nou 00:40:06 x Seruer started, Redis version 1.3.17 
Dr Nou 00:40:06 
Dr Nou 00:40:07 
Dr Nou 00:40:12 
Dr Nou 00:40:17 

Nou 

Nou 

Nou 

Nou 

Nou 

Nou 

Nou 

Nou 


The server is now ready to accept connections on port 6379 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 
clients connected slaves), 450736 bytes in use 


1 1 1 1 1 1 I 1 D D D 1 1 1 TI 1 1 1 1 E 


H 
0 
H 
0 
H 
0 
0 
0 
0 
0 
0 
0 
0 
0 
0 
0 
0 
H 
0 


图 10-3 启动 Redis 
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如 果 Windows 装 有 防火 墙 ， 需 要 允许 防火 墙 通 过 6379 端口 ， 和 否则 PHP 连接 将 被 终止 ， 
如 图 10-4 所 示 。 


É Windows 安全 警报 | DEE 


ei Windows 防火 墙 已 经 阻止 此 程序 的 部 分 功能 


| inwa 防火 墙 已 阻止 所 有 专用 网 络 上 的 redis-server. exe 的 某 些 功能 。 


CR AFN): 
发 布 者 企 ): ”未 知 
13): C:\redis-2. 0. 0-rc¢\redis-server. exe 


人 允许 redis-server. exe 在 这 些 网 络 上 通信 : 
(d 专用 网 络 ， 例 如 家 庭 或 工作 网 络 仆 ) 


许 程序 通过 防火 墙 有 何 风险 ? 


| 允许 访问 他) 


图 10-4 Redis 通过 防火 墙 


至 此 ，Windows 下 的 Redis 服务 端 怠 安装 完成 了 。redis-cli.exe 是 命令 行 管理 终 疹 ， 直 接 
使 用 redis-cli 命令 运行 即 可 。 

2. Æ Linux 上 安装 Redis 

作为 PHP 开发 人 员 ， 需 要 重点 掌握 在 Linux MA PZN. F FR CentOS 
6.0 JFE, EHTA Redis 的 安装 及 配置 过 程 。 

(1) 下 载 Redis 

读者 可 以 在 http:/www.redis.io/download 网 址 下 载 最 新 的 Redis 安装 包 ， 这 里 下 载 的 版 本 
为 2.4.17， 命 令 如 下 。 


[root0-]# mkdir Jopt/data/ 


[roctt+]#} cd Zopt/data 


[root@~]# wget http://redis.googlecode.com/files/redis-2.4.17.tar.gz 

(2) 安装 Redis 

下 载 完 成 后 ， 束 可 以 安装 Redis 了 。 在 安装 前 逢 要 确保 当前 平台 上 已 经 安 状 编译 工具 
(可 参考 8.4.2 节 )， 安 闭 过 程 如 下 。 


[root@~]# tar ZXVE redis-2.4.17.tar.gz 


[root@~]# cd redis-2.4.17 
[root@~]# make 


[root@~]# cd srce 


[root@~]# make install 


执行 完 make install 安装 命令 后 ， 安 装 脚 本 将 会 提示 如 下 信息 。 
mkoir =p /usr/local/bin 


cp -pf recdis=server /usr/local/bin 

cp -Df recdis=-benchmark /usr/local/bin 

cp -pE recdis=cli /usr/local/bin 

cp =p recis=check=cdumo /usr/local/bin 
co -Df redis-check-aof /usr/local/bin 


以 上 信息 提示 Redis 已 经 安装 完成 ， 现 在 需要 创建 /usr/local/redis/bin 目录 ， 并 将 Redis 
户 闪 复制 到 该 目录 下 ， 便 于 操作 与 管理 。 这 里 将 Redis MAL EMRA m) 都 复制 到 
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/usr/local/redis 目录 ， 过 程 如 下 。 


[root@~]# mkdir -p /usr/local/redis/bin 

[root@~]# mkdir -p /usr/local/redis/etc 

root@~]# cp -pf redis-benchmark /usr/local/redis/bin 
root@~]# cp -pf redis-cli /usr/local/redis/bin 
root@~]# cp -pf redis-check-dump /usr/local/redis/bin 
root@~]# cp -pf redis-check-aof /usr/local/redis/pbin 
root@~]# cp -pf redis-server /usr/local/redis/bin 
[root@~]# cp -pf ../redis.conf /usr/local/redis/etc 


操作 完成 后 ， 可 以 在 /usr/local/redis/ 目 录 下 查看 到 相关 文件 。 全 此 ，Redis 的 安装 就 完成 
了 了 ， 只 需要 启动 Redis 服务 即 可 。 

(3) 局 动 Redis 

同样 ， 在 Linx 系统 下 局 动 Reids 时 需要 指定 配置 文件 。 这 里 只 需要 将 配置 项 
daemonize 设 为 true 即 可 ， 该 选项 可 以 把 Redis 推送 到 后 侣 运行， 过程 如 下 。 


[root@~]# vi /usr/local/redis/etc/redis.conf 


# By default Redis does not run as a daemon. Use 'yes' if you need it. 

# Note that Redis will write a pid file in /var/run/redis.pid when daemonized. 

daemonize yes 
保存 redis.conf 配置 文件 ， 局 动 Redis 服务 ， 过 程 如 下 。 
[root@~]# /usr/local/redis/bin/redis-server /usr/local/redis/etc/redis.conf 
局 动 完成 后 ， 可 以 通过 符 看 Redis 主 程序 是 否 存 在 ， 以 确定 Redis 是 否 安 装 成 功 。 
[root@~]# [root@bogon bin]# bstree |grep redis 

| -redis-server---2* [{redis-server}] 
Redis 默认 使 用 6379 通信 端口 ， 可 以 使 用 netstat -anput 命令 检测 Redis 是 人 否 运行 成 功 。 
WRF A TKE, mi IYF 6379 im OME. 


10.3.2 ”安装 Redis 扩展 


Redis 是 一 个 完善 、 功 能 强大 且 易 用 的 NoSQL 数据 库 ， 之 所 以 能 够 被 广泛 使 用 ， 与 
Redis 官方 提供 稳定 、 可 靠 的 API 分 不 开 。Redis 项 目 提供 了 包括 Java、C#、C++、PHP、 
JavaScript 等 所 有 主流 语言 的 API， 读 者 可 以 在 http:/www.redis.io/clients 网 址 下 载 到 相应 的 
安装 包 。 对 于 PHP 而 言 ， 只 下 载 PHP 类 文件 或 者 为 PHP 解释 引擎 安装 扩展 模块 即 可 。 显 然 
安装 扩展 模块 更 加 方便 。 这 里 将 使 用 phpredis 扩展 模块 ， 下 载 地 址 为 https://github.com/ 
nicolasff/phpredis。 

1. Æ Windows TE% phpredis 扩展 

在 Windows 下 安装 PHP 扩展 比较 简单 ，phpredis 分 为 32 位 版 本 及 64 位 版 本 ， 安 装 时 
需要 注意 选择 。 下 面 介 绍 安 疙 过 程 。 

读者 可 以 在 https://github.com/nicolasff/phpredis/downloads 下 载 相应 版 本 的 Windows 扩 
E Q Amt PH 版 本 号 对 应 )， 这 里 下 载 的 版 本 为 php_redis-5.3-ve9-ts-73d99c3e.zip。 解 
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开 压 缩 包 ， 将 得 到 php redis.dll 扩展 模块 ， 将 该 文件 复制 到 PHP 扩展 目录 ， 例 如 
D: eh 然后 编辑 php.ini 配置 文件 ， 加 入 extension=php redis.dll 配置 项 ， 如 以 下 
代码 所 示 。 


963 extension=php El 
904 extcension=php slite3.dl 
905 9053 pextensLon=phap Sybase Ct CLI 
906 pextension=po TLC CLI 

907 extension=php xmlLrpe , CLl 
906 extension=php memcache , cdi 
989 extension=php_redis.dll 

990 

991 [PECL] 

992 extension=php mLNG -CUI 
EE E 
EE 


保存 php.ini 配置 文件 ， 重 启 Web 服务 嚣 ， 至 此 整个 安装 过 程 完成 了 。 要 检测 是 否 安装 
成 功 ， 可 以 通过 phpinfo 函数 输出 信息 ， 查 看 是 否 存 在 Redis， 如 图 10-5 所 示 。 


图 10-5 Redis 配置 信息 


2. 在 Linux 下 安装 phpredis 扩展 

使 用 PHP 内 置 的 phpize 工具 ， 可 以 很 方便 地 为 PHP 引擎 添加 扩展 模块 。 接 下 来 将 继续 
使 用 phpize 安装 Redis 扩展 模块 。 读 者 可 以 在 https://github.com/nicolasff/phpredis/downloads 
下 载 phpredis 扩展 模块 。 详 细 的 安装 过 程 如 下 。 

(1) 下 载 phpredis 
[root@~]# wget http://soft.beauty-soft.net/lib/phpredis-master.tar.gz 


[root@~]# tar zxvf phpredis-master.tar.gz 


[root@~]# cd phpredis-master 

(2) 使 用 phpize 工具 安装 phpredis 

EG PHP 安装 路 径 为 /usr/local/php/， 那 么 phpize 足 径 就 位 于 /usr/local/php/bin/phpize， 评 
细 安 朔 过 程 如 下 。 


[root@~]# /usr/local/php/bin/phpize 
[root@~]#./configure --with-php-config=/usr/local/php/bin/php-config 


[root@~]# make 


[root@~]# make install 
Installing shared extensions: /usr/local/php/lib/php/extensions/no-debug-non-zts-20090626/ 
可 以 看 到 ，phpredis 的 模块 安装 与 Memcache 模块 的 安装 并 没有 多 大 区 别 。 安 装 完 成 后 
Redis 扩展 模块 的 路 和 任 为 /usr/local/php/lib/php/extensions/no-debug-non-zts-20090626/redis.so。 
TIE, Linux 环境 下 的 phpredis 扩展 模块 安装 完成 。 
(3) 在 配置 文件 中 加 入 Redis 扩展 
最 后 只 需要 在 php.ini 配置 文件 中 加 入 Redis 扩展 模块 即 可 (php.ini 路 径 可 以 通过 phpinfo 


278 O O 


第 10 章 网 站 静态 化 


RAEE, M ue H a kl ete 月 录 ， 例 如 /sr/local/php/etc/)。 


BEE Heer lib php ertensions/ no debug non zte 20090626 


808 extension ~ memcache. soit 
S09 extension = "peo mysal. so“ 
810 extension = "redis.so" 


保存 php.ini 文件 ， 重 局 Web 服务 器 ， 至 此 整个 安装 过 程 完 成 了 。 要 检测 是 否 安装 成 
功 ， 可 以 使 用 phpinfo KAA ETE Redis 信息 。 


10.3.3 ”测试 Redis 


验证 Redis 数据 库 及 phpredis 扩展 模块 是 否 正确 运行 ， 最 和 直接 有 效 的 办 法 是 使 用 PHP 代 
人 码 进 行 测试 。 下 面 将 使 用 一 段 简 单 的 代码 对 Redis 进行 测试 ， 代 人 码 如 下 所 示 。 
<?php 
Şredis = new Redis () ， 
Sred isgs=->connect ("192.168215 6379); 
Sredils >set (test "helll vor ldai Tmi 
echo $redis=->0GeT ("test") g 


RS 


10.3.4 Redis 缓存 


Redis 与 Memcached 不 同 ，Redis 不 是 专 为 缓存 而 设计 的 ， 绥 存 只 是 其 中 的 一 项 功能 。 
本 章 主要 介绍 缓存 应 用 ， 下 面 将 结合 ThinkPHP 提供 的 Redis 绥 存 驱动 ， 实 现 数据 缓存 。 详 
细 的 Redis 使 用 在 第 16 章 将 会 有 详细 介绍 ， 在 此 读者 只 需要 当 作 绥 存 服务 器 理解 即 可 。 

1. 开启 Redis 缓存 

在 ThinkPHP 中 使 用 Redis 绥 存 张 动 与 使 用 其 他 缓存 张 动 并 没有 区 别 ， 开 有 人 员 只 需要 
在 配置 文件 中 配置 好 与 Redis 相关 的 配置 项 ， 在 使 用 时 直接 切换 即 可 ， 与 Redis 相关 的 配置 
项 如 下 。 

> REDIS_HOST: Redis 服务 器 IP 地 址 ， 例 如 127.0.0.1。 

> REDIS PORT: Redis 服务 右 开 放 痛 口 ， 默 认 6379。 

> DAIA_CACHE_TIME: FAUTE, AA 0 不 限制 。 

此 外 ， 还 可 以 直接 通过 Cache 中 间 件 中 的 getInstance 方法 初始 化 Redis 配置 信息 ， 如 以 
下 代码 所 示 。 


public function incex () {1 


SCache = Cache::getInstance ( 
'Redis',array ( 
V7 ne koea p 
“DOSE 三 六 外 92 16086-2215"; 
// 端 口 
EE 
// 绥 存 过 期 时 间 
"timeout"=>3600 
II: 


} 

初始 化 完成 后 ， 就 可 以 直接 使 用 Cache 类 保存 缓存 了 ， 如 以 下 代码 所 示 。 
SCache->set ('name', 'ThinkPHP'); JJ ame i 

$value = $Cache->get ('name'); // 获取 绥 存 的 name 数据 
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如 上 述 代 人 码 所 示 ， 这 里 的 set 及 get 方法 不 是 phpredis 模块 中 的 方法 (phpredis 模块 也 包 
F set 及 get 方法 )， 而 是 Cache 绥 存 中 间 件 所 提供 的 缓存 统一 处 理 方法 。 

2. 使 用 Redis 缓存 Session 

使 用 Redis 作为 缓存 服务 占 在 使 用 方式 上 与 Memcached 一 样 的 ， 旋 者 可 参考 表面 介绍 的 
Memcached 实战 内 容 部 分 。 接 下 来 将 通过 创建 一 个 Session 驱动 ， 进 一 步 认识 Redis 数据 组 
存 功能 。 

首先 在 ThinkPHP/Extend/Driver/Session 目录 创建 Session 驱动 ， 并 命名 为 SessionRedis. 
class.php， 代 人 码 如 下 所 示 。 


<?php 


class SessionRedis { 
// 有 效 时 间 
protected) $11ferime=s!" "yg 
// 前 级 
protected) Sprefi="" ; 
/ /Memcached DIR 
protected shander; 


x Qparam string SsavePath 


x Qparam mixed Ş$sessName 


n 
public function open ($savePath, ŞsessName) { 
Şthis->lifeTime = C" SESSION BXRPIRE")S 
$this->prefi = CE O TONT PREET A: 
sredis = new Redis (); 
redi conneee EE EE EE EE EE 
es el 
Şthis->hander=$redis; 


Cerc CVS 


}else{ 
return false; 

} 
} 
/** 
EE 
ee en 
OE tp Fore we pe EE TE YEE ep EV OE ES A EE EE EE 


十 二 
KÉ 
Eeer 
EE EE 


EE 


} 


280 O O 


第 10 章 网 站 静态 化 


publie function read ($sessIiD) 4 
şdata=$this->hander->get (Şthis->prefi.ŞsessID); 
return Gdata: 


* Qparam string $sessID 


* Qparam String Ş$sessData 


SEH? 
public function write(S$sessID,S$sessData) { 
Şexpire = $Şthis=>lifteTime; 


return $this=->hander=>s6tex(Sthis=->prefi. $sessID, Sthis >11iferime, $SsessData) ; 


z 
publie function cestroy ($sessID) 4 
return Schis=>hander=>delete(SthLs=>prefi, SSESSID) 7 


-i 


publie function ge ($sessMazbireTime) 4 
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TEIN crues 


x Qparam string SsavePath 


x Qparam mixed Ş$sessName 


a 
Public function execute) | 
gesso ser tere ee Leen il e oven 

array (&$this,"close"), 
array (&$this,"read"), 

eStchis, write”), 

tHEhis, destroy") ， 

array (eSthis, "ge™) ) s 


array 


array 


( 
( 
( 
( 


} 


} 
在 使 用 Session 时 不 和 需要 修改 任何 代码 ， 只 需要 将 Session 类 型 改 为 Redis 即 可 ， 如 以 下 
代码 所 示 。 


"oseTON Vee Re 
人 
SEE 


10.4 ”静态 缓存 


剖面 介绍 的 绥 存 都 是 针对 数据 库 查 询 的 ， 事 实 上 ThinkPHP 还 文 持 静态 文件 缓存 。 静 态 
文件 缓存 可 以 根据 控制 及 控制 器 动作 生成 缓存 ， 生 成 缓存 后 页 面 束 等 于 完全 静态 化 (与 生成 
HTML 类 似 )， 在 绥 存 有 效 期 内 页 面 数 据 不 再 依赖 后 台数 据 库 。 

页 面 静态 绥 存 非常 适用 于 信息 数据 量 大 、 页 面 更 狐 少 、 访 问 量 大 的 页 面 ， 例 如 Blog、 新 
闻 系 统 等 。 静 态 缓存 是 ThinkPHP 核心 功能 之 一 ， 使 用 时 不 需要 额外 配置 环境 ， 也 不 需要 额 
外 引入 扩展 ， 只 需要 在 配置 文件 中 开启 HTML CACHE ON 选项 即 可 。 静 态 缓存 共 由 两 部 分 
组 成 :静态 绥 存 配置 、 项 态 缓存 规则 。 只 有 正确 地 开启 或 配置 ， 项 态 绥 存 才能 够 正确 运行 ， 
下 面 分 别 介绍 。 


10.4.1 静态 缓存 配置 


静态 绥 存 配置 涉及 绥 存 的 生效 时 间 、 路 径 存 放 、 组 存 扩 展 名 称 等 。 静 态 绥 存 配置 与 常见 
的 项 目 配置 项 相 类 似 ， 其 中 HTML CACHE ON 配置 项 是 静态 缓存 的 总 开关 ， 静 态 缓存 配置 
选项 如 表 10-2 所 示 。 
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表 10-2 缓存 缓存 配置 项 


默 认 项 
HTML CACHE ON 静态 绥 存 生成 状态 false 
HIML_CACHE_TIME 静态 缓存 有 效 时 间 ， 以 秒 为 单位 ， 为 0 时 表示 不 限时 间 60 
HTML FILE SUFFIX 静态 缓存 文件 名 后 缀 .html 
HTML PATH 静态 文件 存放 目录 ./appName/Html 
HTML CACHE RULES 静态 缓存 规则 array() 


需要 注意 的 是 ，HTML _PATH 默认 路 径 为 项 目 根 目录 下 的 HTML 文件 夹 ， 如 果 该 文件 
夹 不 存在 ， 系 统 将 会 自动 创建 。 这 就 意味 着 项 目 存放 目录 (appName) 需要 赋予 可 读 可 写 权 
限 〈 在 非 Windows 下 )， 但 更 理想 的 做 法 是 修改 HTML _PATH 配置 值 ， 让 该 值 指向 一 个 可 读 
可 与 的 全 局 缓存 目录 中 ， 例 如 Runtime 目录 。 


10.4.2 ”静态 缓存 规则 


静态 绥 存 规则 是 静态 绥 存 生成 中 最 关键 的 配置 。 毅 态 绥 存 规则 与 URL 路 由 规则 有 些 类 
似 ， 这 两 者 的 核心 都 是 基于 正则 匹配 的 。 静 态 绥 存 规则 格式 如 下 。 

路 人 径 匹配 规则 =>array( 文 件 绥 存 规则 ', SRRA au HI "附加 规则 ") 

静态 绥 存 规则 共 由 两 部 分 组 成 : 路 人 径 匹 配 规 则 及 文件 缓存 规划。 文件 绥 存 规则 又 分 为 妾 
规 配置 及 附加 配置 ， 下 面 分 别 介绍 。 

1. 路 径 匹 配 规则 

路 径 匹 配 规则 决定 了 缓存 的 区 域 ， 系 统 共 提 供 了 4 种 路 径 匹 配 规则 《全 部 小 号 )， 分 别 
为 全 局 动作 绥 存 规则 、 控 制 右 绥 存 规则 、 控 制 融 动作 绥 存 规则 以 及 全 站 绥 存 规则， 下 和 而 分 列 
介绍 。 

(1) 全 局 动作 绥 存 规则 

全 局 动作 绥 存 规则 是 根据 URL 请 求 ， 匹 配 项 目下 的 所 有 动作 并 生成 相应 的 静态 文件 。 
全 局 动作 绥 存 规则 是 最 常用 、 最 方便 的 一 种 廊 态 绥 存 规则， 规则 格式 如 以 下 。 

'index'=>array('index','60') 

标 粗 的 即 为 全 局 动作 绥 存 规则 。 其 中 index 即 为 控制 右 动 作 ， 表 示 在 URL 请 求 到 index 
动作 时 ， 系 统 将 生成 index.htm 于 态 绥 存 文件 。 这 里 使 用 的 是 全 局 配置 ， 意 味 看 只 要 用 户 访 
问 项 目下 的 index 动作 名 称 ， 不 管 来 目 哪 个 控制 器 ， 都 会 生成 天 态 绥 存 。 

(2) 控制 舌 绥 存 规则 

控制 如 缓存 规则 能 够 六 能 地 根据 控制 右 生 成 绥 存 分 类 ， 并 且 将 分 类 (栏目 ) 下 的 所 有 动 
作 都 生成 静态 缓存 ， 这 个 过 程 是 无 需 开 发 人 员 干 预 的 ， 系 统 会 日 动 根据 URL 请 求 完成 缓存 
生成 与 读 取 。 控 制 右 缓存 规则 格式 如 下 。 

'index:'=>array('index/{:action} {1d}",'600') 

标 粗 的 即 为 控制 器 绥 存 规则 。 其 中 indes 表示 控制 器 名 称 ,“:” 分 隔 符 是 全 局 动作 绥 存 
规则 与 控制 袁 绥 存 规则 的 区 别 所 在 《注意 为 碳 文 冒 亏 )。 表 示 当 URL 请 求 到 index 控制 左下 
的 任意 一 个 动作 《页 面 )， 系 统 将 目 动 创建 或 读 取 毅 态 缓存 文件 。 
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(3) 控制 项 动作 绥 存 规则 

控制 袁 动 作 缓存 规则 是 全 局 动作 缓存 规 则 的 子规 则 ， 能 够 将 绥 存 规则 区 域 进一步 缩小 。 
控制 器 动作 绥 存 规则 根据 URL 请 求 中 的 控制 磺 及 动作 名 称 ， 匹 配 绥 存 规则 中 的 范围 。 如 采 
匹配 结 末 成立 ， 则 生成 或 读 取 绥 存 ， 否 则 放 径 缓存 操作 。 探 制 左 动 作 缓 存 规则 格式 如 下 。 

' index:user'=>array(' {1d}',0) 

标 粗 的 即 为 控制 器 动作 绥 存 规则 。 其 中 indes 表示 控制 器 名 称 ，user 表示 index 控制 器 
下 的 user 动作 。 需 要 注意 的 是 ， 通 凋 情 况 下 只 需要 对 有 意义 的 控制 右 动 作 做 况 态 化 处 理 即 
可 ， 所 以 上 述 格式 只 适合 于 常见 的 控制 占 动 作 廊 态 绥 存 规则 ， 并 不 适合 空 控制 费 及 空 动作 。 

(4) 全 站 绥 存 规则 

爹 站 绥 存 规则 使 用 的 是 泛 匹 配 模 式 ， 如 来 将 该 规则 配置 到 项 目 配 置 文 件 ， 则 将 项 目下 的 
所 有 控制 占 动 作 痢 做 前 态 化 处 理 ; 如 来 将 全 站 绥 存 规则 配置 到 全 站 公共 配置 文件 
Cconfig.inc.php)， 则 无 论 访问 网 站 哪个 页 面 ， 都 将 做 毅 态 化 处 理 。 全 站 绥 存 规则 格式 如 下 。 

一 >afrray({9 SERVER.REQUEST URllmdain 

上 述 缓存 规则 表示 根据 当前 页 面 生 成 或 读 取 毅 态 文 件 ， 并 使 用 md5 函数 序列 化 文件 名 
称 。 全 站 绥 存 规则 是 一 种 贫 柳 的 匹配 模式 ， 过 多 的 使 用 会 极 大 地 消耗 服务 器 资源 ， 从 而 让 绥 
存 效果 得 不 偿 失 ， 如 果 在 数据 更 新 频繁 的 网 站 上 使 用 ， 有 可 能 造成 不 可 预料 的 运行 错误 。 

2. 文件 缓存 规则 

文件 缓存 规则 是 系统 在 处 理 缓存 时 的 一 种 文件 命名 规则 ， 它 直接 影 啊 到 缓存 的 保存 名 
称 、 有 效 期 等 。 文 件 缓存 规则 共 由 绥 存 规则 、 缓 存 有 效 期 及 附加 规则 构成 。 通 单 情况 下 ， 只 
需要 配置 缓存 规则 即 可 ， 系 统 共 文 持 5 种 缓存 规则 ， 下 和 面 分 别 介绍 。 

(1) 使 用 PHP 全 局 变量 

PHP 内 置 了 许多 全 局 变量 ， 例 如 $_SERVER、$_SESSION、$_COOKIE 等 。 文 件 缓存 规 
则 人 允许 开发 人 员 直 接 在 缓存 规则 内 调用 PH 内 阜 的 全 局 变量 ， 最 终生 成 的 绥 存 文件 将 以 全 
局 变量 的 值 作为 缓存 文件 名 ， 如 以 下 代码 所 示 。 

l 文章 列表 

v Enter description Here 
= 
rT 
cookie ("cache", "article"); // 当 面 页 面 缓存 名 称 


Şobj=M ("Article"); 
Srows=$0b]=>0rder ("id dese") Eeer 


ŞStnis=>assign ("List $rows) g 
Stnhis=>dLsplay (|); 


} 
EREZZE, OD COOKIE 全 局 变量 获取 cache 值 即 可 ， 如 以 下 代码 
所 示 。 


TREE 
'HTML CACHE TIME'=>86400, 
CTM CACHE RULES =e arra, 
AER E ea ne 


8 
如 上 述 代码 所 示 ， 在 缓存 规则 中 使 用 变量 ， 需 要 使 用 定 界 符 “{ }”。 如 来 变量 为 数组 ， 
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使 用 “.” 连 接 和 从 匹配 数组 下 标 键 。 与 视图 标签 一 样 ， 文 件 缓存 规则 允许 使 用 函数 对 变量 值 做 
一 步 处理 ， 变 量 值 与 图 数 之 间 使 用 “| ”分 陋 待 ， 如 以 下 代码 所 示 。 
aa CACHE ONT true, 
T CACHE TIME ss DÉADO, 
(DM, CACHE BULESi ss artau) 
'index:article'=>array('{$ COOKIE.cache|md5}'), 


E 

上 述 代 码 表示 使 用 md5 函数 加 密 $_COOKIE['‘cahce’]。 

(2) 使 用 框架 变量 

为 了 方便 开发 人 员 得 到 当前 环境 信息 ，ThinkPHP 框架 本 身 就 内 置 了 许多 对 外 公开 
的 变量 或 第 量 ， 开 友人 员 可 以 直接 将 这 些 变 量 舱 入 到 文件 缓存 规则 里 ， 常 见 的 框架 变量 
如 下 。 

> {:app}: 获得 当前 应 用 名 称 ， 即 入 口 文件 定义 的 APP NAME 常量 值 。 

> {:group} 获取 当前 应 用 分 组 名 称 ， 没 有 分 组 时 将 显示 为 空 。 

> {:module} 获取 当前 控制 占 名 称 。 

> f{:action} 获取 当前 动作 名 称 。 

框架 变量 之 间 可 以 组 合 使 用 ， 也 可 以 单独 使 用 。 如 果 多 个 变量 之 间 使 用 “/” 分 隔 符 ， 
系统 将 根据 变量 前 后 顺序 生成 相应 的 目录 结构 ， 最 后 的 变量 值 为 绥 存 文件 名 。 如 以 下 代码 
所 示 。 


A CACHE ONT true, 
'HTML CACHE TIME'=>86400, 
"HTML CACHE RULES'=- array! 
indexiarticle = array | module / kac tioni Corie cache lmda>) Ee 


E 
上 述 配置 规则 将 以 控制 器 名 称 及 动作 名 称 生 成 缓存 文件 存放 路 径 ， 并 以 md($_COOKIE 
[cahce"]) 作 为 缓存 文 件 名 。 

(3) 使 用 $_GET 参数 值 

缓存 规则 是 以 URL 请 求 为 基础 的 ， 所 以 可 以 直接 使 用 $_GET[' 变 量 名 "] 的 方式 取得 参数 
值 。 在 默认 的 视图 模板 引擎 中 ， 要 获取 URL 参数 值 ， 只 需要 使 用 凤 参 数 名 } 方 式 获取 即 可 。 
同样， 在 绥 存 规则 中 也 是 允许 这 样 使 用 的 。 如 以 下 代码 所 示 。 

'HTML CACHE ON'=>true, 

'HTML CACHE TIME'=>86400, 


'HTML CACHE RULES'=> array ( 
'index:article'=>array('{:module}]/{:action}/article ($p}'), 


) ， 

如 上 述 代码 所 示 ，{$p} 变量 与 $_GETI[‘p’] 是 相等 的 。 参 数 p 是 框架 内 置 的 一 个 分 页 变 
量 ， 该 值 保 存 看 当前 分 页 页 人 码 。 

(4) 使 用 函数 

使 用 函数 是 一 种 比较 灵活 及 强大 的 文件 缓存 规则 ， 这 里 的 函数 不 仅 文 持 PHP 内 置 的 函 
数 ， 还 文 持 框架 或 项 目 中 的 函数 〈 包 括 自 定义 函数 、 扩 展 函 数 等 )。 使 用 函数 时 不 再 使 用 
“$” 构 量 声 明 符 号 ， 而 是 使 用 “|” 分 隔 符 作为 标识 ， 如 以 下 代码 所 示 。 

NE 


'HTML CACHE TIME'=>86400, 
'HTML CACHE RULES'=> array ( 
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'index:article'=>array ('{|time}'), 


) 1 

(5) 混合 使 用 

混合 使 用 是 在 前 面 介绍 的 4 种 缓存 规则 的 基础 上 ， 实 现在 一 条 绥 存 规则 内 同时 运用 多 条 
文件 缓存 规则 ， 如 以 下 代码 所 示 。 

'HTML CACHE ON'=>true, 

"HTML CACHE TIME'=>86400, 


'HTML CACHE RULES'=> array ( 
'index:article'=>array ('{:module}/{:action}_{|time}'), 


E 
3. 文件 缓存 附加 规则 
文件 绥 存 附加 规则 是 文件 缓存 规则 中 一 种 扩展 机 制 ， 通 第 使 用 目 定义 函数 实现 文件 缓存 
规则 。 事 实 上 ， 附 加 规则 不 是 必需 的 ， 因 为 在 前 面 介 绍 的 $ 种 文件 缓存 规 则 中 第 4 种 就 可 以 
使 用 日 定义 函 数 。 所 以 文件 缓存 附加 规则 通常 用 于 后 期 维护 。 如 以 下 代码 所 示 。 
'HTML CACHE ON'=>true, 
'HTML CACHE TIME'=>86400, 


HTML, CACHE RULES > array! 
Made aree e EE 0 em ES 


8 
附加 规则 中 指定 的 temp 函数 ， 只 需要 在 common.php 公共 函数 库 中 实现 即 可 ， 代 码 如 
Fro, 


function temo (Svar) 1 


return "aW Syars 


} 
BAREK ERR Em EEI ~” PRR, Aan “~article.htm”. Æ 
实际 应 用 开发 中 ， 可 以 使 用 目 定 义 函数 ， 实 现 功 能 强大 的 附加 规则 。 


10.5 ”小结 


本 半 针 对 大 数据 绥 存 进行 了 详细 讲解 ， 结 合 ThinkPHP 内 置 的 绥 存 驱动 、Session 驱动 ， 
使 得 第 三 方 大 型 缓存 服务 器 能 够 无 颖 地 集合 到 MVC 开发 模式 中 。 

本 章 一 开始 对 系统 内 置 的 Cache 缓存 中 间 件 进行 了 深入 介绍 ， 因 为 只 有 理解 缓存 中 间 
件 ， 在 使 用 S ARRERA BEN ERIE T ATF 

通过 目 定 义 扩展 ， 使 得 数据 缓存 、Session 绥 存 变 得 容易 ， 通 过 使 用 大 型 的 数据 缓存 服 
务 器 ， 不 仅 使 缓存 速度 变 快 ， 同 时 也 让 数据 变 得 更 加 安全 及 稳定 ， 还 有 效 地 解决 了 Session 
跨 域 问题 ， 所 以 在 真实 应 用 开发 中 ， 如 果 条 件 允 许 ， 建 议 读者 多 答 试 使 用 绥 存 服务 器 处 理 绥 
存 及 Session, 

ThinkPHP 文 持 多 种 绥 存 服务 器 (数据 库 )， 由 于 篇 幅 关 系 ， 本 章 重 点 对 主流 的 
Memcached 及 Redis 进行 深入 介绍 ， 读 者 在 学 习 过 程 中 可 以 尝试 编写 其 他 绥 存 数据 库 驱 动 。 
最 后 还 深入 介绍 了 基于 IO 操作 的 静态 缓存 ， 全 面 实现 网 站 静态 化 。 
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内 容 提要 


扩展 是 衡量 MVC 框架 是 否 灵 活 及 强大 的 重要 标志 ， 一 套 完善 的 MVC 框架 必须 提供 
扩展 机 制 。ThinkPHP 的 扩展 机 制 是 非常 灵活 和 高 效 的 ， 在 前 面 的 内 容 介绍 中 已 有 涉及 。 
ThinkPHP 的 扩展 机 制 根 据 功 能 与 作用 共 分 为 8 大 类 ， 前 面 已 涉及 的 有 驱动 扩展 及 类 库 扩 
RK. 

除 此 之 外 系统 还 提供 模式 扩展 、 控 制 器 扩展 、 模 型 扩展 、 行 为 扩展 、 函 数 扩展 、 增 强 扩 
展 等 。 本 章 首 先 将 对 系统 内 置 的 扩展 进行 全 面 介 绍 ， 在 此 基础 上 进一步 介绍 增强 扩展 机 制 ， 
实现 自 定 义 扩 展 。 通 过 本 章 的 学 习 ， 读 者 不 仅 可 以 全 面 了 解 系统 内 置 的 扩展 机 制 ， 还 能 深入 
掌握 PHP 高 级 使 用 方法 。 


学 习 目 标 


了 解 扩展 和 概念。 

掌握 文件 上 传 扩展 的 使 用 。 

掌握 网 络 通 信和 扩展 的 使 用 。 

掌握 ThinkPHP 内 置 常见 扩展 类 库 的 使 用 。 
理解 ThinkPHP 行为 (拦截 器 )。 
掌握 行为 扩展 的 使 用 。 
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11.1 使 用 扩展 


ThinkPHP 的 扩展 机 制 灵活 局 效 ， 在 使 用 方式 上 不 需要 复杂 的 配置 。 根 据 功能 与 作用 不 
同 ， 扩 展 的 使 用 方式 也 有 所 不同 ， 但 主要 分 为 类 库 导 入 方式 及 文件 配置 方式 。 接 下 来 和 和 完 介 
绍 扩展 的 分 类 ， 帮 助 读 者 理解 系统 内 置 的 扩展 原理 ， 然 后 再 介绍 扩展 的 配置 。 


11.1.1 扩展 的 分 类 


表面 介绍 过 系统 共 文 持 8 大 类 扩展 ， 这 些 扩展 是 系统 原生 文 持 的 ， 但 扩展 类 库 文件 并 非 
系统 内 置 ， 需 要 开发 人 员 手 动 下 载 ， 读 者 可 以 在 http:/www.thinkphp.cn/extend/ 找 到 相关 信息 
及 扩展 文件 。 下 面 分 别 介绍 。 

1. 模式 扩展 

传统 意义 上 ， 一 般 使 用 MVC 进行 开发 吏 是 指 网 站 开发 ， 所 谓 的 网 站 开发 需要 由 视 网 、 
控制 器 、 模 型 组 成 。 但 是 随 大 互联 网 的 发 展 ， 很 多 情况 下 网 站 只 是 其 中 的 一 个 单元 ， 共 同 构 
造 运营 平台 的 还 有 客户 端 、 开 放 API CRM (客户 管理 系统 )、OA (办 公 自 动 化 系统 ) 等 。 
如 何 将 这 些 单元 或 设备 整合 到 一 起 ， 是 网 站 开发 人 员 需 要 面 对 的 问题 。ThinkPHP 的 模式 扩 
展 很 多 情况 下 就 适用 于 开发 这 些 模块 ， 以 简化 开发 难度 ， 提 高 运行 速度 。 系 统 共 文 持 7 种 模 
式 扩 展 ， 下 面 将 分 别 对 最 常用 的 儿 种 模式 进行 介绍 。 

(1) 人 简洁 模式 

系统 在 初始 化 时 ， 默 认 使 用 标准 模式 运行 。 标 准 模 式 最 典型 的 模块 有 视图 、 数 据 库 
ORM 等 。 人 简洁 模式 就 是 将 标准 模式 进行 了 简化 ， 使 得 项 目 改 变 运 行 模 式 ， 人 处 于 简洁 模式 的 
项 目 具 有 以 下 特性 。 

> 不 支持 模块 (控制 器 ) 分 组 。 

> 不 载 入 视图 引擎 (但 可 以 手动 调用 )。 

> 不 文 持 数 据 模 型 (数据 映射 、CURD、 了 字段 处 理 等 )， 只 能 使 用 原生 SQL 操作 数据 库 。 

> RRA MySQL Zhi EIK 

> 不 载 入 多 语言 处 理 模块 。 

> 不 支持 Dispatch 路 由 。 

> 不 能 使 用 系统 内 置 扩 展 ， 但 可 以 使 用 目 定 义 扩展 。 

人 简洁 模式 通过 减少 模块 的 载 入 ， 从 而 使 框 染 更 加 轻 量 级 。 由 于 没有 视图 引擎 ， 所 以 简洁 
模式 非常 适合 开发 类 似 于 Ajax 后 台 、JavaScript 输出 等 简单 的 应 用 。 定 义 人 简洁 模式 ， 只 需要 
在 项 目 入 口 文件 定义 MODE NAME 常量 值 为 Thin 即 可 ， 如 以 下 代码 所 示 。 

define ( 'MODE NAME See 

(2) 精简 模式 

精简 模式 是 一 种 比 人 简洁 模式 提供 更 多 功能 的 应 用 开 友 模式。 精简 模式 非常 适合 开发 小 型 
的 留言 板 、 向 导 系 统 等 。 精 简 模式 的 MODE NAME 常量 值 为 Lite， 如 以 下 代码 所 示 。 

define ( 'MODE NAME' SE 

与 简洁 模式 相 比 ， 精 简 模 式 在 人 简洁 模式 的 基础 上 增加 了 如 下 儿 项 特性 。 

> 载 入 系统 默认 的 视 网 引 擎 。 
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> 文 持 Dispatch 路 径 功 能 。 

> HIE DIE, cé, 

> 文 持 没有 回调 结果 的 CURD RE. 

(3) 命令 模式 

在 Linux Shell 编程 中 ， 通 常 使 用 Java、C、C++、Perl 编写 命令 脚本 。PHP 作为 一 种 
Web 动态 脚本 ， 通 常用 于 网 站 编程 ， 但 Zend 引擎 引入 了 命令 行 模 式 ， 人 允许 开发 人 员 下 接 将 
PHP 编译 为 Shell 脚本 ， 在 Linux 系统 下 只 要 安装 了 PHP 解释 器 ， 束 能 够 像 其 他 编程 语言 一 
样 ， 使 用 PHP 作为 Shell 脚本 。 

命令 行 模式 就 是 运行 在 Shell ( 即 Linux 终端 命令 ) 编程 模式 下 的 ， 开 发 人 员 只 需要 在 
Shell 脚本 中 以 传 参 的 方式 给 MVC 动作 传递 命令 ，PHP 脚本 即 会 执行 相关 操作 《例如 操作 数 
据 库 、IO 操作 等 )。 命 令 行 模 式 的 MODE NAME 常量 值 为 cli， 如 以 下 代码 所 示 。 

define('MODE NAME', 'cli'); 

命令 行 模式 非常 适合 于 Shell fE, MRES, WANIE drop íT kret 
的 任务 。 在 传 参 时 需要 使 用 Linux 标准 的 命令 参数 。 例 如 /usr/local/bin/php index action id 4 表 
示 请 求 index 模块 action 方法 ， 并 传递 参数 id， 参 数值 为 4。 

(4) AMF 模式 

ZendAMF 是 一 球 由 Zend 公司 开发 并 定义 的 数据 传输 协议 ， 用 于 满足 flex, flash 编程 时 
与 PHP 进行 大 数据 交互 的 需要 。 数 据 交 互 常 用 的 技术 有 HTTPService、WebService 和 
RemoteObject 。 这些 技术 都 是 基于 RCP (远程 过 程 调 用 协议 ) 的 。HTTPService 和 
WebService 使 用 XML 数据 格式 ，RemoteObject 使 用 ZendAMF 数据 格式 。 

ZendAMF 数据 是 基于 数据 流 的 ， 在 传输 时 需要 序列 化 与 反 序 列 化 ， 所 以 无 论 性 能 还 是 
传输 量 ，ZendAMF 都 比 XML 更 有 优势 。 使 用 AMF 模式 将 让 应 用 更 加 适合 ZendAMF 编 
程 ， 实 现 与 flex 高 效 互 动 。 在 ThinkPHP 中 ， 要 修改 项 目 为 AMF 模式 ， 只 需要 在 入 口 文件 
中 修改 MODE NAME 值 为 amnf 即 可 。 如 以 下 代码 所 示 。 


define ('MODE NAME', 'amf'); 

将 运行 模式 改 为 amf 后 ， 还 需要 在 配置 文件 中 定义 允许 被 RemoteObject 调用 的 控制 
器 ， 如 以 下 代码 所 示 。 

'APP_AMF_ACTIONS'=>' Index, User, Shop', 

关于 了 RemoteObject 的 使 用 ， 访 者 可 以 参考 相关 的 flex 开发 资料 ， 在 此 不 做 深入 介绍 。 

(5) RPC 模式 

RPC 通信 是 一 种 应 用 非常 广泛 的 数据 区 换 技术 ， 例 如 HITPService 、 WebService 
(SOAP), WCF 等 都 是 基于 RPC 来 通信 的 。RPC 能 够 文 持 多 种 数据 格式 ， 例 如 XML, 
WSDL, Json, ZendAMF 等 。 可 以 说 RPC 是 客户 疹 与 服务 端 最 通用 、 也 是 最 可 对 的 一 种 通 
信 协 议 。 在 ThinkPHP 中 ， 定 义 RPC 通信 和 模块 与 Amf 类 似 。 首 先 在 配置 文件 中 修改 
MODE NAME 常量 值 为 phprpc， 如 以 下 代码 所 示 。 

define ('MODE NAME', 'phprpc'); 

然后 在 项 目 配置 文件 中 定义 公开 可 调用 的 控制 器 即 可 ， 如 以 下 代码 所 示 。 

'APP_PHPRPC ACTIONS'=>' Index, User, Shop', 

将 项 目 改变 phprpe 模式 ， 可 以 让 项 目 专注 于 处 理 rpm 调用 ， 而 不 需要 处 理 Web 实现 ， 
例如 WebService 处 理 等 ， 非 营 适 合 开 发 网 站 API。 
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2. 控制 器 扩展 
一 个 控制 器 束 是 一 个 文件 类 ， 例 如 UserClass。 开 发 人 员 可 以 通过 控制 器 扩展 ， 实 现 为 
控制 右 添 加 附加 功能 ， 例 如 客户 端 判 谈 、 权 限 检验 等 。 控 制 硕 扩 展 需 要 使 用 _initialize 方法 接 
口 实现 ， 如 以 下 代码 所 示 。 
<?php 
class JIndexAction extends Action { 
/ /扩展 接口 
EE EE 
ŞUSer=M ("User"); 
Şusername=session ("username"); 
Şpassword=session ("password"); 
See ee 
iE EE 
$this->error (" 你 还 没 登 录 ， 请 登录 后 再 操作 ") ; 


} 

public function veriiy() 
1moo0Ort ("ORG UCLL, Image™) 5 
Image: :buildīImageVerify(); 

} 

publice funCtion incex () 4 


E 
Scnis=>display(); 


} 
R> 


3. 模型 扩展 

模型 提供 了 对 数据 表 和 直接 操作 的 作用 ， 通 过 模型 扩展 ， 可 以 增强 模型 的 功能 。 系 统 本 吴 
内 置 了 多 种 扩展 模型 ， 如 AdvModel 〈 高 级 模型 )、RelationModel 等 。 与 控制 器 一 样 ， 模 型 
的 扩展 接口 也 是 _initialize 方法 ， 开 发 人 员 可 以 使 用 _initialize 方法 实现 模型 接口 的 初始 化 ， 
调用 基础 类 库 或 者 函数 每。 扩展 模型 允许 使 用 模型 内 置 的 增强 方法 ， 如 _before insert、 
_after insert 等 ， 从 而 实现 更 高 级 的 CURD 操作 。 可 扩展 的 模型 增强 方法 如 表 11-1 所 示 。 


表 11-1 CURD 操作 接口 


作用 说 明 

_before insert($data,$options) 保存 数据 前 接口 add 
after insert($data,$options) 保存 数据 后 接口 add 
_before update(&$data,$options) 更 新 前 接口 save 
_after_update($data,$options) 更 新 后 接口 save 
_after delete($data,$options) 删除 后 接口 delete 
after select(&$data,$options) 查询 多 条 数据 后 接口 select 
after find(&$data,$options) 查询 单条 数据 后 接口 find 


WK 11-1 所 示 ， 参 数 $data 表示 传 入 或 返回 的 数据 信息 “数组 类 型 )， 参 数 $options 表示 
返回 当前 操作 的 模型 名 称 及 表 前 级 “数组 类 型 )。 其 中 更 新 前 操作 及 更 狐 后 操作 还 可 以 使 用 
_facade($data) 代 和 登 。CURD 扩展 接口 的 使 用 如 以 下 代码 所 示 。 
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<?php 
class UserModel extends Modell 
/大 大 
* add 前 操作 接口 
ce Jee eforeyims eer 
wy 
Ee Eon EE EE 
sdatal"password"]=md5 ($datal[l"password"]); 
return S$data; 


} 
E 


4. 行为 扩展 

行为 相当 于 请 求 管理 器 ， 行 为 可 以 根据 URL 动作 请 求 ， 在 执行 动作 前 执行 的 一 系列 系 
统 扩展 功能 。 例 如 URL 路 由 检测 、 模 板 定 位 、 令 牌 生成 等 。 系 统 共 内 置 了 8 大 行为 ， 如 下 
所 示 。 


checkRoute: 检测 URL 路 由 。 
LocationTemplate: 定位 模板 文件 路 径 。 
ParseTemplate: 调用 模板 解释 引擎 。 
ShowPageTrace: 显示 页 和 面 Trace 信息 。 
ShowRuntime: 显示 系统 运行 时 间 。 
TokenBuild: 初始 化 表单 令 脾 。 
WriteHtmlCache: 生成 缓存 。 
ReadHtmlCache: 读 取 缓存 。 
行为 的 基 类 为 Behavior， 人 允许 开发 人 员 通 过 继承 的 方式 扩展 行为 。 关 于 行为 的 使 用 ， 本 
革 后 面 将 详细 介绍 ， 在 此 读者 只 需要 理解 行为 的 作用 即 可 。 
5. 函数 扩展 
通过 目 定 义 函 数 可 以 实现 函数 的 扩展 ， 函 数 扩 展 在 前 面 草 和 内 容 中 已 经 多 次 介绍 ， 谈 者 
应 该 已 经 掌握 ， 在 此 不 下 过 多 介绍 。 
6. 增强 扩展 
通过 自 定义 的 类 库 扩展 可 以 实现 系统 本 映 不 内 置 的 功能 ， 例 如 发 送 电 子 邮 件 、GD 绘图 
等 。 这 也 是 本 章 重 点 介绍 的 内 容 。 


VW WWW WW VY 


N 


11.1.2 ”模板 引擎 扩展 


模板 引擎 扩展 是 一 种 比较 特殊 的 扩展 ， 它 不 允许 开发 人 员 对 扩展 进行 日 定义 ， 必 须要 严 
格 按 照 第 三 方 模板 引 黎 提供 的 处 理 方式 设计 模板 。 也 残 是 说 如 末 改 变 模板 引擎 ， 模 板 必须 重 
新 设计 。ThinkPHP 内 置 了 两 种 模板 引擎 扩展 〈Smarty 及 Template)。 要 更 改 模板 引擎 ， 只 需 
要 在 配置 文件 中 指定 TMPL_ENGINE_TYPE 配置 项 即 可 ， 默 认 如 下 。 

CON 

这 里 需要 注意 的 是 ， 虽 然 系统 提供 了 和 便 板 扩展 机 制 ， 但 只 能 完成 基础 的 模板 解释 操作 ， 
对 于 便 板 原 有 的 内 壮 扩 展 并 不 文 持 ， 例 如 在 模板 中 使 用 目 定 义 函 数 、 闫 库 、 绥 存 等 。 但 我 们 
可 以 通过 扩展 控制 右 的 方式 实现 ， 如 以 下 代码 所 未。 
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<?php 
class PubAction extends Action { 
publie Semartys 
publie function init smarty () 1 
// 加 载 Smarty 模板 扩展 
vendor DE NEE EE Class- php™) p 
Şthis->smarty=new Smarty (); 
// 默 认 模 板 目 录 
sehis >smarty >templatedir APP PATH. T /Dpl/T: 
this >smarty >Compile dir-APE PAI. /RUuntime/ Teompilest: 
1 a EE 
PEhiss Smarty config r APP TEAT Cone 
// 缓 存 目录 
Ehis >smarty cache dir APP PATH. /Roaneime/ TCachei: 
Sthig=->smarty->caching=fralse; 
// 模 板 定 界 符 
SE ere leren elei 
Grois- Ee delimiter] > 


} 
SE 


如 上 述 代码 所 示 ， 通过 创建 一 个 PubAction 的 公共 控制 项 ， 在 该 控制 希 中 实现 Smarty Ri 
板 引 擎 的 初始 化 ， 项 目下 的 所 有 控制 器 均 继 承 于 该 控制 器 ， 通 过 这 种 方式 就 可 以 实现 全 功能 
的 Smarty 模板 扩展 ， 在 设计 模板 时 开发 人 员 可 以 使 用 Smarty AEKA DBE WA PRH 
Dr, 


public function 1ncex() 
ŞUser=M ("User"); 
Srows=S$User->select (); 
Sms > nm na 
this >smarty -template dir-AFP PATH. 1/Tpl/ndex'; 
Sthals=>smarty=>assign ("LLST SLOWS) $ 
ŞSthals=>smarty=>assign ("Title", "feof"); 
Şthis=>smarty=>cisplay (" index neml’)s 


} 
index 动作 对 应 的 index.html 模板 文件 如 以 下 代码 所 示 。 


<IniEnml > 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
EE EE 

</head> 

<body> 

<div data-role="content"> 


<ul data-dividertheme="b" data-theme="c" data-role="listview" data-inset="true"> 
ll foreach from=Slist LCEMSCOwWS ==> 
<li><a href=" #UserPage" ><!--{Ş$rows[|"UserName"] }--></a></li> 
<!--{/foreach}--> 

</ul> 

</div> 

</body> 

</html> 


关于 Smarty 模板 引擎 的 使 用 ， 本 书 第 13 革 将 会 深入 介绍 ， 在 此 读者 只 需要 掌握 Smarty 
模板 引擎 的 村 入 即 可 。 
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11.2 网络 操作 


网 络 操 作 是 网 站 开发 中 最 第 见 的 功能 ， 例 如 文件 上 传 、 数 据 采 集 等 。 本 市 将 深入 介绍 
ThinkPHP 对 网 络 操作 的 文 持 ， 使 用 系统 内 置 的 封 姜 英 库 ， 可 以 局 效 、 稳 定 地 开发 网 络 通信 
应 用 。 此 外 本 节 还 深入 讲解 Nginx 内 置 的 文件 上 传 功 能 ， 实 现 文件 上 传 进度 实时 显示 。 


11.2.1 文件 上 传 


PHP 内 置 了 非 钊 强大 、 易 用 的 文件 上 传 模 块 ， 使 得 过 去 需要 额外 扩展 才能 实现 的 文件 上 
fk, 在 PHP 中 只 需要 使 用 一 个 简单 的 move uploaded file 函数 即 可 实现 。ThinkPHP 内 置 了 
文件 上 传 扩 展 类 UpdateFile， 该 类 库 对 move uploaded file 函数 进行 了 高 度 封 装 ， 实 现 了 上 
传 文件 类 型 检测 、 上 传 文件 大 小 检测 、 上 传 文件 压缩 等 重要 功能 。 下 面 将 深入 介绍 。 

1. UpdateFile 类 库 

虽然 PHP 提供 了 move uploaded file 函数 ， 能 够 方便 地 实现 文件 上 传 功 能 。 但 一 个 完善 
的 文件 上 传 步 骤 至 少 需要 包括 文件 大 小 检测 、 类 型 检测 等 ， 然 后 才能 执行 move uploaded ` 
file 上 传 ， 如 以 下 代码 所 示 。 


<?php 

Suploaddir = ”,/uploads/" ; 

SEN = Leder 
Suploadfile = Suploaddir  Şrilename, 

if ((ọ EPITESI file] [tzie] T024) -I0 


a ee 
} 
$type=$ FILES['file']['tmp name']; 
stype=strstr ($type,"."); 
EE EES 
exit ("文件 类 型 正确 "); 
} 


Tn ee, Eet EE a ae a aaee a 
exit ("文件 上 传 上 传 ")， 
}else{ 


S T V 
} 
> 


在 MVC 开发 中 ， 这 些 操作 均 被 封 六 为 功能 类 。 文 件 类 型 、 大 小 等 使 用 类 成 员 属 性 定 
义 ，UpdateFile 类 库 是 ThinkPHP 内 置 的 文件 上 传 扩 展 类 ， 使 用 时 需要 额外 引入 。 如 以 下 代 
公所 未 。 


Import ( ORG. Net- Uploack ile”) s 
UpdateFile 类 是 实例 类 ， 使 用 时 需要 实例 化 。 一 个 最 简单 的 上 传 功能 只 需要 调用 upload 
方法 即 可 ， 如 以 下 代码 所 示 。 


sup load ~ ner U load le): // 实例 化 上 传 类 
Upload -maxSize = 31458; // 设置 附件 上 传 大 小 
Siload -allonExts =~ arra (jp). // 设置 附件 上 传 类 型 
Şuüupload--savePath -~ "/Public/upload"; // 附件 上 传 目 录 
Supload-zupload (ii: / /执行 上 传 
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如 上 述 代 人 码 所 示 ，UpdateFile 扩展 类 中 提供 了 maxSize, allowExts 等 实用 属性 ， 这 些 属 
性 可 以 对 文件 大 小 、 类 型 进行 检测 ， 防 止 用 户 随 意 上 传 文件 ， 给 系统 造成 安全 隐患 。 
UpdateFile 提供 了 众多 文件 上 传 处 理 属性 ， 如 表 11-2 所 示 。 


表 11-2 UpdateFile 扩展 类 成 员 属 性 


S 认 值 
常规 选项 
maxSize 上 传 文件 最 大 字 节 数量 -1， 不 限制 
savePath 上 传 文件 存放 路 径 ， 需 要 可 读 写 权限 UPLOAD PATH 
SaveRule 上 传 文件 保存 规则 ， 需 要 唯一 性 ， 例 如 time) C (UPLOAD FILE RULE”) 
hashType 上 传 文件 的 验证 方式 md5_file 
autoCheck 是 否 自 动 对 上 传 文件 进行 大 小 、 类 型 、 合 法 性 检测 true 
uploadReplace 存在 同名 文件 是 否 窗 盖 false 
allowExts 允许 上 传 文件 后 级 ， 关 联 数组 类 型 array()， 个 限制 
allowTypes 允许 上 传 文件 类 型 ， 关 联 数组 类 型 array()， 不 限制 
缩 略 图 
thumb 是 否 生 成 缩 略 图 false 
thumbMaxWidth 缩 略 图 宽度 ， 多 个 之 间 使 用 “,” 隔 开 
thumbMaxHeight 缩 略 图 高 度 ， 多 个 之 间 使 用 “,” 隔 开 SS 
thumbPrefix 缩 略 图 前 级 thumb 
thumbSuffix Haig AI Ja 2 z 
thumbPath 缩 略 图 保存 路 径 室 ， 使 用 savePath 值 
thumbFile 指定 缩 略 图 保存 名 称 z 
thumbRemoveOrigin HE AA MA e E R IR vm false 
THK 
autoSub 是 否 局 用 子 目录 存放 上 传 文件 false 
subType 子 目 录 生 成 规则 ， 可 选 的 值 有 hash、date hash 
hashLevel 子 目 录 生 成 层次 l 


2. Image 类 库 

UpdateFile 文件 上 传 类 对 图 像 文件 处 理 是 基于 Image 类 库 实 现 的 ， 同 时 Image 类 也 是 一 
个 对 外 公开 的 扩展 类 位 于 ORG/Util 目录 )。 开 发 人 员 可 以 直接 调用 Image 图 像 处理 类 直接 
对 上 传 图 片 进行 进一步 处 理 。Image 闫 包括 了 毅 态 成 员 方 法 及 实例 成 员 方 法 ， 静 态 方 法 使 用 
时 不 需要 实例 化 。 下 面 将 对 钊 用 的 成 员 方法 进行 介绍 。 

(1) getImageInfo 

getImagelnfo 方法 用 于 获取 图 像 文件 综合 信息 (基于 PHP 内 置 的 getimagesize 函数 实 
现 )， 方 法 形式 如 下 。 

static function getImageInfo ($img) 

参数 img 表示 完整 的 图 片 文件 路 径 。 一 旦 获取 成 功 getImagelInfo 方法 将 会 返回 图 像 宽 
度 、 高 度 、 大 小 等 。 如 以 下 代码 所 示 。 


public function imoex()! 
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rietsen e (ORG, Utcl, Image™) 
$data=Image: :getImageInfo ("./Public/Images/123.jpg"); 
qun (odata); 

} 

(2) water 

water 方法 能 够 生成 图 片 水 印 ， 方 法 形式 如 下 。 


static public function water ($source, $water, $savename=null, $alpha=80) 

参数 source 表示 需要 加 入 水 印 的 原 图 片 ， 参 数 water 表示 水 印 图 片 ， 只 支持 jpg、png 图 
片 格式 ; 参数 savename 表示 水 印 合 成 后 的 图 片 文件 ， 参 数 alpha 表示 水 印 透明 度 。 

(3) buildString 

buildString 方法 可 以 将 字符 串 转 换 为 图 片 ， 方 法 形式 如 下 。 


static function buildString ($string, Srdgb=array()， Şfilename='', $type='png', 


Ssdisturb=1, Sborder=true) 
参数 string 表示 需要 转换 的 学 符 串 ， 不 文 持 中 文 ， 参数 rgb 表示 图 像 宇 体 的 颜色 值 ， 例 
如 array("0","0""0 几 表示 黑色 字体 ;参数 filename 表示 输出 文件 名 ; 参数 type 表示 图 像 类 
型 ， E EE 参数 disturb 表示 图 像 干扰 点 ， 共 支持 4 个 值 : 0 无 干扰 
点 )、1 点 干扰 )、2 线 干扰 )、3 (复合 干扰 )， 参 数 border 表示 是 人 否 添 加 网 像 边框 。 


11.2.2 Nginx 文件 上 传 进度 


网 站 文件 上 传 多 数 都 是 使 用 HTTP 提交 实现 的 ， 与 蜗 面 程序 使 用 Socket 等 上 传 方式 不 
EJ, HTTP 通信 是 一 种 异步 单线 程 的 通信 方式 ， 所 以 处 理 文 件 实时 上 传 进度 一 直 是 网 站 开发 
的 难题 。 接 下 来 将 使 用 Nginx 上 传 模 块 ， 实 现 文 件 上 传 及 文件 上 传 进度 显示 功能 。 在 这 之 
有 前， 首先 了 解 目前 主流 的 文件 上 传 技术 。 
1. 文件 上 传 进度 
ye 
上 传 模块 等 ， 下 面 分 别 介 
(1) 浏览 绒 插 件 
使 用 浏览 器 插件 是 当前 大 型 网 站 所 普 遇 采用 的 上 传 技 术 ， 国 内 的 视频 网 站 、QQ EA 
di (ad mW" 等 都 是 基于 浏览 器 插件 技术 实现 文件 上 传 的 。 利 用 E 浏览 器 Acivex 技术 可 
以 实现 将 时 和 面 级 的 程序 租 入 到 网 页 中 ， 网 站 束 可 以 充分 利用 操作 系统 强大 的 特性 ， 轻 松 地 实 
现 多 线程 文件 上 传 、 进 度 显 示 、 文 件 夹 上 传 、 文 件 转换 等 。 
多 线程 处 理 ， 在 上 传 文件 过 程 中 还 能 够 实现 队列 上 传 、 T 
然 Acivex 是 微软 IE 独 有 的 技术 ， 但 主流 浏览 费 都 提供 了 扩展 功能 ， 通 过 安装 扩展 ， 同 样 能 
够 实现 类 似 功 能 。 浏 览 器 插件 技术 并 非 全 是 优点 ， 也 有 缺点 ， eer 
> 由 于 浏览 占 插 件 本 质 上 是 一 种 加 而 软件 ， 这 束 意 味 看 开 友 人 员 需 要 额外 尝 握 浏览 高 
扩展 开发 技术 ， 这 对 于 普通 的 PHP 开发 人 员 而 言 是 极 大 的 挑战 。 
> 对 于 下 浏览 器 而 言 ， 不 能 使 用 插件 的 方式 实现 浏览 器 扩展 ， 需 要 使 用 Activex 技 
术 。 出 于 安全 考虑 ， 通 常 系统 已 将 浏览 器 级 别 调 到 最 高 ， 这 种 安全 级 别 下 操作 系统 
是 禁止 和 目 动 安 装 Activex 控件 的 。 一 些 杀 毒 软件 同样 提供 了 相同 的 拦截 机 制 。 所 以 滞 
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见 的 视频 网 站 部 需要 用 户主 动 安装 客户 澳 ， 或 者 浏览 右 控 件 。 对 于 中 小 型 网 站 而 
言 ， 这 种 用 户 体验 是 很 难 被 认可 的 。 
> 随 看 客户 端 浏览 右 或 操作 系统 的 升级 ， 搬 件 或 Activex 必须 要 提供 相应 的 升级 更 新 ， 
这 无 疑 增 加 了 网 站 运营 成 本 。 
> 浏览 帮 插 件 技术 不 能 应 用 于 手机 、 和 平板 电脑 等 移动 终 病 。 
(2) Flex 
Flex 是 Flash 的 演进 技术 ,与 Jquery、JavaFx 一 样 是 一 种 富 互 联网 技术 (RAI)， 使 用 
Flex 技术 能 够 轻 多 地 创建 出 极 具 美观 、 动 感 的 网 站 。 国 内 的 Web 游戏 多 数 都 是 基于 Flex K 
现 的 ， 但 由 于 性 能 问题 ， 以 及 HIML 5 的 问世 ， 近 年 来 Flex 的 发 展 受 到 了 一 些 挑 战 。 
但 在 文件 上 传 方面 ，Flex 一 直 是 最 通用 、 主 流 的 技术 。 使 用 Fles 实现 网 站 文件 上 传 ， 
不 仅 可 以 实现 进度 显示 、 多 文件 上 传 、 文 件 夹 上 传 等 功能 ， 而 且 由 于 元 分 利用 操作 系统 网 络 
处 理 功能 ， 能 够 轻松 地 实现 大 文件 上 传 、 断 点 上 传 每 。 此 外 还 可 以 结合 Adobe Fms IK 
术 实 现 视频 实时 采集 等 局 级 应 用 。 
本 质 上 Flex 也 是 一 种 浏览 器 插件 技术 ， 但 由 于 Flash 的 普及 ， 使 用 Flex 技术 开发 的 网 站 
儿 平 不 需要 用 户 额 外 安装 浏览 器 扩展 ， 这 对 于 用 户 而 言 无 疑 是 友好 的 。 但 Fles 也 有 缺点 ， 
归纳 如 下 。 
> Flex 是 一 种 由 Adobe 掌控 的 技术 ， 不 像 HTML, JavaScript 是 由 W3C 定义 的 技术 ， 
所 以 需要 开发 人 员 对 Adobe 开发 拉 术 有 深入 理解 ， 例 如 网 络 通信 、 文 件 处 理 等 。 
> Flex 使 用 Flash Play 演 染 ，Flash Play 是 一 种 浏览 堪 插 件 ， 近 年 来 多 次 被 曝 严 重 安全 
漏洞 。 用 于 网 站 文件 上 传 ， 需 要 把 安全 问题 考虑 在 内 。 
> 虽然 多 数 浏 览 器 都 支持 Flex 技术 ， 但 由 于 性 能 问题 ， 移 动 终端 并 不 支持 Flex. 
相似 的 技术 还 有 JavaFx 以 及 Silverlight 等 。 这 些 技术 共同 点 均 需 要 额外 安装 浏览 器 扩 
展 ， 虽 然 功 能 强大 ， 但 用 户 体 验 并 非 民 好 。 
(3) Ajax 技术 
Ajax 是 由 JavaScript, Xml, CSS 等 综合 构成 的 技术 ， 在 易 用 性 及 通用 性 上 是 其 最 大 优 
Ko Ajax 最 大 的 之 点 是 能 够 实现 网 页 局 部 无 刷新 操作 ， 从 而 改进 文件 上 传 体 验 。 
传统 的 网 页 文件 上 传 需要 使 用 表单 提交 到 PHP 处 理 页 面 ， 再 由 PHP 处 理 表 单 中 的 文件 
数据 。 处 理 完 成 后 再 返回 结束 ， 开 发 人 员 需 要 在 返回 结 采 后 将 页 面 导 航 到 成 功 处 理 页 面 。 
使 用 Ajax 能 够 将 文件 上 传 步骤 改 为 无 刷新 操作 。 这 一 过 程 用 户 始 终 感 党 不 到 曾经 离开 
过 当前 页 面 ， 但 事实 上 文件 上 传 表单 已 经 被 后 从 处 理 完毕 ， 并 返回 结 琳 。 为 了 让 用 户 感 觉 到 
文件 在 上 传 ， 开 有 人 员 还 可 以 使 用 动画 图 请 模拟 上 传 进度 ， 在 一 定 程度 上 提高 用 户 体验 。 
由 于 Ajax 本 质 上 是 JavaScript， 所 以 包括 移动 终端 在 内 的 许多 设备 ， 都 能 够 很 好 地 文 持 
Ajax。 只 要 该 设备 浏览 器 支持 文件 表单 域 (IOS 6.0、Windows CE, Windows RT 等 均 已 文 
持 )， 就 可 以 很 好 地 实现 路 平台 。 上 所 以 现在 很 多 中 小 型 网 站 都 广泛 地 使 用 Ajax 处 理 文件 上 
传 。 同 样 ，Ajax 也 存在 一 些 缺 点 ， 人 简单 归 纳 如 下 。 
> Ajax 虽然 实现 了 无 刷新 提交 表单 ， 但 本 质 上 还 是 HITP 网 页 提交 ， 后 台 的 PHP 脚本 
处 理 能 力 决定 了 文件 上 传 速度 ， 最 终 直 接 决 定 用 户 上 传 体验 。 
> Ajax 的 文件 上 传 进度 是 使 用 动画 图 片 模拟 的 ， 事 实 上 并 不 能 真实 反应 文件 上 传 进 
度 ， 如 果 上 传 的 文件 比较 大 ， 这 种 动画 模拟 并 不 能 真正 有 效 改善 用 户 体 验 。 
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> Ajax 通常 只 能 上 传 较 小 的 文件 ， 例 如 用 户头 像 、 邮 件 附 件 等 。 

需要 注意 的 是 ，Ajax 只 能 实现 同一 网 站 内 的 文件 上 传 ， 不 能 实现 跨 网 站 上 传 。 

(4) 服务 嚣 上传 模 块 

服务 器 上 传 模块 是 一 种 用 于 改进 Web 上 传 低 效 的 技术 ， 例 如 脚本 运行 超时 、 内 存 洲 
出 、CPU 负载 过 高 等 。 服 务 器 上 传 模块 是 一 种 高 效 的 文件 上 传 技 术 ， 包 使 用 操作 系统 底层 
的 网 络 处 理 技 术 ， 实 现 多 线程 的 文件 上 传 功能 。 目 前 Lighttp 以 及 Nginx 都 提供 有 相应 的 文 
件 上 传 扩 展 ， 根 据 笔者 的 实际 测试 ，Nginx 上 传 扩 展 在 效率 上 起 码 比 传统 的 PHP 脚本 提高 
LJ, 

Nginx 文件 上 传 扩展 ， 虽 然 不 能 像 浏 览 器 扩展 那样 能 够 实现 文件 夹 上 传 、 断 点 续 传 等 高 
级 功能 。 但 是 Nginx 上 传 扩 展 文 持 文 件 集群 存放 《 文 持 存 放 数 据 云 )、 文 件 实 时 进度 显示 、 
文件 加 密 等 功能 ， 所 以 Nginx 上 传 扩 展 能 够 适用 于 各 类 型 网 站 文件 上 传 的 需要 ， 并 能 够 有 效 
改善 用 户 体 验 。 接 下 来 将 重点 介绍 Nginx 文件 上 传 扩 展 功能 。 

2. Nginx 文件 上 传 

Nginx 用 于 处 理 文 件 上 传 的 扩展 模块 共有 两 个 ， 分 别 为 nginx upload module (EEX 
件 ) 及 nginx_uploadprogress_module《〈 显 示 文 件 进度 )。Nginx 在 编译 时 默认 并 没有 加 入 这 两 
个 扩展 模块 ， 需 要 开发 人 员 手 动 进行 编译 。 下 和 面 首先 介绍 模块 的 安 疙 。 

(1) 安装 文件 上 传 模块 

默认 情况 下 ，Neginx 并 没有 安装 nginx upload module 及 nginx uploadprogress module 模 
块 ， 读 者 可 以 使 用 nginx -V 检测 是 否 存在 该 模块 。 

[root@~]# /usr/local/nginx/sbin/nginx -V 

接 下 来 手动 重新 编译 nginx upload module 及 nginx uploadprogress module 模块 ， 这 里 
使 用 的 操作 系统 为 CentOS 6.0, Nginx 版 本 为 1.3.8。 首 先 下 载 软 件 包 。 


[root@~]# mkdir /opt/sdal/sft/ 


[root@~]# cd /opt/sdal/sft/ 


[root@~]# wget http://soft.beauty-soft.net/lib/nginx upload module-2.2.0.tar.gz 


[root@~]# wget http://soft.beauty-soft.net/lib/nginx uploadprogress module-0.9.0.tar.gz 


[root@~]# wget http://soft.beauty-soft.net/lib/nginx-1.3.8.tar.gz 


ICH 20 Hl, Zou A ENR Nginx 依赖 库 。 


[root@~]# yum -y install gcc dcc-c++ autoconf libjpeg libjpeg-devel libpng libpng- 


devel freetype freetype-devel libxml2 libxml2-devel zlib zlib-devel glibc glibc-devel 
glib2 glib2-devel bzip2 bzip2-devel ncurses ncurses-devel curl curl-devel e2fsprogs 
e2fsprogs-devel krb5 krb5-devel libidn libidn-devel openssl openssl-devel openldap 
openldap-devel nss ldap openldap-clients openldap-servers 

接着 使 用 tar 解压 下 载 到 的 源 代 码 包 。 
[root@~]# tar zxvf nginx upload module=-2.2:0.tar:gz 


(root: tar zxvf nginx uploadporogress module-0:9.:0.tar:gz 


[root@~]# nginx-1.3.8.tar.gz 
nginx upload module 及 nginx uploadprogress module 扩展 模块 不 需要 安装 ， 只 需要 在 编 
YE Nginx 时 谎 加 上 即 可 ， 如 果 已 经 安 朔 过 Nginx， 同 样 也 需要 重新 编 详 。 
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[root@~]# ./configure --user=www --group=www --prefix=/usr/local/nginx --with-http 
stub status module --with-http ssl module --add-module=/opt/sdal/sft/nginx upload module- 
2.2.0 --add-module=/opt/sdal/sft/masterzen-nginx-upload-progress-module-a788dea 

通过 前 面 的 步骤 ， 接 下 来 上 只 需要 执行 make 即 可 。 


[root@~]# make 


[root@~]# make install 

通过 闻 面 的 步骤 ， 现 在 nginx upload module 及 nginx uploadprogress module 模块 已 经 被 编 
译 进 Nginx 了 。 需 要 注意 的 是 ， 这 里 只 是 介绍 nginx upload module 及 nginx Uploadprogress 
module 模块 安装 ， 方 便 接 下 来 的 学 习 ， 如 果 访 者 全 新 安装 Nginx GEEN mE), m Aí 
安装 PHP 及 PHP-FPM， 可 参考 本 书 第 1 章 12.20. 

(2) BC Nginx 

LAAR, MÆ Nginx 已 经 文 持 文 件 上 传 模 块 ， 接 下 来 只 需要 在 配置 文件 中 配置 
上 传 模块 ， 即 可 实现 文件 上 传 。 

nginx upload module 的 上 传 原理 也 是 通过 HTTP 进行 提交 的 ，nginx_ upload module 
能 够 自动 对 表单 中 的 附件 进行 处 理 ， 开 发 人 员 只 需要 提交 相应 的 表单 给 Nginx 即 可 。 整 
个 过 程 PHP 均 不 直接 参与 文件 处 理 ， 上 只 需要 负责 提交 数据 及 处 理 提 区 后 的 数据 即 可 。 为 
了 方便 操作 ， 这 里 将 在 Nginx Server 市 点 中 配置 节点 ， 用 于 统一 处 理 表 单 提 交 请 求 。 如 
以 下 代码 所 示 。 


location /upload { 


upload pass meteai, 
upload store /home/wwwroot/file 1; 
uploadi store access user: ry 
woles aet EE nom a oa ae y 
aeee ae oa a ea a lee alel nane e EE EE ENEE 
EE EE Ra EE 
eee ao See TOT 7 1 Van A a 
ENEE EE O70 
upload mas E Eet sum Leer Ee 
uploadi pass args on; 
Erack uploads Toxo 0s, 

} 

been e bettel 


rewrite “(+*) /test.php Last: 


| 

上 述 代 但 各 项 配置 参数 含义 如 下 : 

> upload pass: 文件 上 传 完成 处 理 通 道 。 通 第 是 一 个 网 站 ， 或 者 PHP 文件 地 址 。 

> upload store: 临时 文件 存放 路 径 。Nginx 运行 用 户 需要 具备 可 庶 可 与 权限 : 参数 1 
表示 存放 目录 散 列 方式 〈Nginx 将 把 文件 随机 存放 到 黎 列 目录 ， 默 认为 0 一 9 )。 

> upload store access: 存放 目录 访问 模块 。 

> upload set form field: 传递 给 upload pass 脚本 的 POST 表单 值 。PHP 可 以 使 用 
$_POST 获取 到 该 数 信 。 

> upload aggregate form field: 变量 集合 。 

> upload pass form field: 传递 给 后 台 的 参数 方式 。 可 以 使 用 正则 。 
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> upload pass_args: 是 否 把 前 并 脚本 请 求 的 参数 传 给 后 病 处 理 程序 。 默 认为 on, 

通过 前 面 的 配置 ， 现 在 已 经 完成 了 nginx upload module 模块 的 配置 。 要 让 Nginx GEN 
实时 返回 上 传 进度 ， 还 需要 配置 nginx uploadprogress module 模块 。 

nginx uploadprogress_module 模块 运行 于 独立 的 线程 ， 在 nginx upload module 上 传 文件 
时 ， 能 够 实时 地 监控 上 传 文件 的 状态 信息 。 开 发 人 员 可 以 使 用 HTTP 的 方式 获取 
nginx_ Uploadprogress_ module 返回 的 数据 ， 这 里 也 是 通过 配置 Server 节点 实现 。 代 人 码 如 下 。 


location “~ /progress 4 


report uploads proxied; 
} 
通过 上 述 配置 ， 现 在 可 以 通过 http:/ DomainName/progress 初始 化 nginx 
uploadprogress_module 扩展 模块 。 这 里 需要 注意 的 是 ，nginx_uploadprogress_module ZO 
根据 x-progress-id 参数 返回 文件 上 传 状态 的 ， 该 参数 值 是 唯一 的 ， 由 上 传 表 单 通 过 x- 
progress-id 参数 提供 。 


location ~ (5 Ee EE 

rewrite ^(,7)/x=7progress=id (\w*) $19?X=-Progress=ID=$2; 
} 
了 最 终 的 Nginx 配置 信息 如 以 下 代码 所 示 。 


user WWW WWW; 
worker processes 1; 
events { 
worker connections 1024; 
j 
heca { 
autoindex on; 
SCHER EE eae EE 
autoindex localtime on; 
default type application/octet-stream; 
sendfile on; 
tcp nopush on; 
cep nocelay on; 
keepaliyes timeout 10; 
gzip On; 
gzip min lengtha lkes 
gzip buffers 4 6k; 
gzip http version 1.1; 
gzip comp level 3; 


gzip types Cext/iecess test/aoml text/plain a pplication x Javascript application zm] 
application/pei appolicacion/rcci apolication/z-perl application/x-tel applicartcion/mswore 
application/vnd.ms-excel application/vnd.ms-powerpoint application/vnd.wap.xhtml+xm1 
image/x-ms-bmp; 
gzip vary on; 
output butters 4 32k; 
ùploacdi progress Json output, 
uploadi progress proxiec! im; 
server í 
listen S07 
server name localhost, 
charset utit=6,; 002312; 
client max bocy size 2000m; 


root /home/wwwroot; 
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error page 405 =200 04105; 
location @405 


{ 


root /home/wwwroot; 


location /upload { 


300 O O 


upload pass Ctesi, 
Zupload pass /test.php?path=uploadfile&a=upload server; 
upload store /home/wwwroot/file 1; 
uiloae STOre access UUSEE 
Belt eet orn eelo ados Tomem aerea aoe e ae 
upload EE EE ee a ae on ws， 
uploadi EE an a aa aAa, 
ploert aA e eee o ae a oea n ene) me a oa e ma 
Der ben, = 
up boa EE | description a 
tupload Pass ormt uel no; 
upload! pass _ args onp; 
Crack UPIS aido proi cdi Os, 
} 
eneen ee ti] 
rewrite A(*)o /test.php last; 
PEO ER 


Leer i onn 
oeo ae heele Hoet areo MOST, 
root /home/wwwroot; 
index index.html index.htm index.php; 


location ~ (.*)/x-progress=ics (\w*) | 
rewrite ^(.*)/xz-progress=Lcs (\w*) S22xX Prooress ID=$2; 
} 
location EE EE 
report uploads proxiecl; 
d 
location ~ \-phps { 
tastegi pass 127-0,.0,. 139000; 
fastegi incex index. pho; 
set -path info “Ti; 
Sete reale pt nam aee oe penae, 
Wi a e a e a a a e | 
3 


} 

Steel Jbg lJpegl png omol] swE)S 4 
root /home/wwwroot; 
access log offs 
expires 30d; 

} 

location ~ »“\» (Jslessl]ico)?$ 4 


root /home/wwwroot; 
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access log ofp 

expires 1h; 
Í 
error page S00 S0250 04 750、 htm], 
location = /50x. neml 1 

root /home/wwwroot; 
} 
NE 
E EE EE 
scene en | Tec， 
include fuero al ng eoni facte oar ams 


} 
配置 完成 后 ， 可 以 使 用 /usr/local/nginx/sbin/nginx -t 测试 Nginx 配置 文件 是 否 正 确 ， 如 果 


配置 无 误 ， 最 后 只 需要 重 局 或 司 动 Nginx 即 可 。 
[root@~]# pkill nginx 


[root@~]# /usr/local/nginx/sbin/nginx 

(3) 测试 文件 上 传 模块 

通过 前 面 的 步 又 ，Nginx 已 经 具备 文件 上 传 功能 了 。 可 以 将 表单 提交 给 upload 模块 处 
理 ， 一 旦 存在 表单 域 ，Nginx 将 目 动 上 传 ， 并 由 progress 返回 状态 信息 。 但 是 如 末 谈 者 百 接 
访问 http:/ DomainNamemupload/， 因 为 不 存在 表单 提交 ，Nginx 将 返回 404 错误 。 这 里 为 了 
便于 测试 ， 前 和 完 在 网 站 中 创建 表单 页 面 ， 如 以 下 代 公 所 示 。 

<html> 


<head> 
Ee EES 


</head> 


<body> 
<form id="upload" enctype="multipart/form-data" action="http://192.168.2.15/ 


upload?X-Progress-ID=123456789" method="post"> 
<input name="file" type="file"/> 
<input type="submit" value="Upload"/> 
</form> 


<div id="uploading"> 
<div id="progress" class="bar"> 
<div id="progressbar">&nbsp;</div> 
di 
el 
<div id="percents"></div> 
</body> 
</html> 
如 上 述 代 码 所 示 ， 其 中 的 表单 提交 地 址 不 再 是 PHP 页 面 地 址 ， 而 是 前 面 配置 的 upload 
服务 恬 模块。 其 中 参数 X-Progress-ID 是 文件 的 唯一 id， 获 取 上 传 进度 时 需要 使 用 。 前 面 介 
绍 过 ，Nginx 在 上 传 时 使 用 0 一 9 的 目录 随机 保存 文件 ， 所 以 在 执行 上 传 前 ， 需 要 手动 在 
/home/wwwroot/file 目录 中 创建 相应 的 目录 。 完成 后 就 可 以 访问 前 面 创建 的 表单 页 和 面 进行 文 
件 上 传 了 。 在 上 传 过 程 中 ， 可 以 通过 http:/ DomainName/progress?X-Progress-ID=xxx 取得 访 
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文件 的 上 传 状态 。 

3. 使 用 Ajax 提交 文件 

通过 页 面 的 介绍 ， 相 信 读 者 已 经 对 Nginx 的 文件 上 传 原理 已 经 有 了 全 面 理解 。Nginx 在 
上 传 文件 时 ， 共 有 两 个 网 址 ， 一 个 用 于 处 理 表单 ， 另 一 个 用 于 返回 上 传 状态 。 

根据 这 个 原 理 ， 只 需要 使 用 Ajax 开 步 提交 表单 ， 同 时 使 用 JavaScript 时 间 触 发 事件 ， 
步 无 刷新 获取 该 文件 的 上 传 进度 ， 并 更 新 页 面 上 的 进度 条 《〈 可 以 使 用 CSS+DIV E 或 图 创 建 )， 
这 样 就 实现 了 文件 上 传 及 文件 上 传 进度 实时 显示 。 

接 下 来 将 使 用 Jquery 提供 的 Ajax 寞 步 上 传 插 件 ， 该 插件 在 传统 的 PHP 上 传 中 使 用 得 非 
向 广泛 。 如 以 下 代码 所 示 。 

<html> 

<head> 


<title>ajaxFileUpload</title> 
<meta htetpo-ecuiy=" Content-Type" content=" text/html} charset= uf 8 /> 


<s5CEript Sre="hnTtEO: // T- ENEE EE 
<script sre="hnttO! // Tt- beaurty=soft- net/ je/ajartilenload. Js"></seript> 
<script type="text/javascript"> 
function upUserHeadimg () { 
vec UUTE = Wiig 
f@EF (var i = 0g 1 < 322 ärt) A 
uule += March. floor (Marth. rancon) *16) - coscring (i6) s 


ş$.ajaxFileUpload ({ 
urls "nrttos//192.168.2,15/uploac?X-Progress=ID=" SE ek 
secureurL: false, 
ELleblementIicl: "ELile", 
dataType: 'post', 
success; functlion (daca, status) 
if (data=="0"){ 
// 没 能 通过 校 验 
alene S Ee Voa 
} 
if (data.indexOf ("error™")>= 0){ 
// 上 传 失败 


}else{ 


error: tunet on (cata, stacus, e) 


alert (e); 


); 
interval = window.setInterval( 
funetion () 1 
getUploadProgress (uuid); 
}, 
20 


return false: 
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} 
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function getUploadProgress (uuid) { 
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var url="htto://192.168.2.15/progress?X-Progress=-ID=" uuid, 


S$ .getJSON (url, function (data) { 
if (data.state=="done"){ 
// 上 传 完成 


$ ("yprogresshar") -css (1wLath: "100%" } ) ; 
var wMath.tloor (0L. L + 100.0 / 1000); 


window.cClearTlimeocout (Interval); 


}else{ 


// 实 时 显示 上 传 进度 


BEE ee 
e EE 


(ir lnporooressbarn) 


} 
</BCript> 
<style type="text/css"> 
.bar { 
widths 300px; 


#progress { 
background: #eee; 
border: lpx solid #222; 
EE EE 20x} 

} 

#progressbar { 
widths 0px 
height: 24px; 
backgrounds, F333; 

j 

</style> 
</head> 


bodvz 
<form id="upload" enctype="multipart/form-data" 
method="post"> 
<input name="file" id="file" type="file"/> 
<input type="submit" value="Upload"/> 
</form> 


<diy ic="uploacdine" > 

<div id="progress" class="bar"> 
<div id="progressbar">&nbsp;</div> 

T 

SE 

<div id="percents"></div> 

</body> 
</html> 


onsulbomi t="renrn 


upUserHeadimg ();" 


上 述 代 码 是 普通 的 HTML 代码 ， 没 有 涉及 任何 PHP 处 理 脚 本 ， 该 者 可 以 直接 在 MVC 


钢 图 中 使 用 。 最 终 上 传 效果 如 图 11-1 ra, 
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图 11-1 文件 实时 上 传 进度 效果 


文件 上 传 完 成 后 ，upload 模块 Cupload pass 定义 的 PHP 脚本 输出 ) 将 返回 上 传 文件 数 
组 信息 ， 这 里 为 了 方便 演示 ， 只 需要 输出 提交 表单 信息 即 可 。testphp 文件 代码 如 下 。 


<?php 
var dump POST); 
2> 


实际 应 用 开发 中 ， 可 以 将 返回 的 数据 保存 到 数据 库 或 缓存 系统 。Nginx 的 上 传 模块 从 设 
计 之 初 束 是 针对 大 型 文件 系统 的 ， 所 以 采用 了 临时 目录 存放 的 特性 ， 在 文件 上 传 完成 后 ， 通 
第 需要 使 用 监控 脚本 将 临时 文件 上 传 到 文件 系统 (例如 MooseFS、HDFS 等 )。 在 实验 时 ， 
该 者 可 以 直接 使 用 PHP 将 文件 移动 到 正式 目录 ， 并 且 使 用 前 面 介 绍 的 Image 类 库 完成 图 片 
截图 、 创 建 水 印 等 常见 操作 。 


O 说明: 如 果 读 者 在 本 地 网 络 中 模拟 上 传 操作 ， 由 于 网 速 过 快 的 原因 ， 可 能 会 出 现 stat 永远 为 done 的 状 
态 ， 即 文件 上 传 完 成 状态 。 解 决 办 法 是 在 真实 的 互联 网 中 进行 上 传 。 


11.2.3 FTP 文件 上 传 


FTP 是 一 种 可 靠 、 稳 定 的 文件 传输 协议 。 与 HTTP 相 比 ， 由 于 FTP 使 用 操作 系统 底层 实 
现 ， 所 以 FTP 能 够 有 效 降低 系统 资源 占用 ， 提 高 上 传 效率 。 接 下 来 将 简单 介绍 PHP 使 用 
FTP 上 传 文件 。 

ThinkPHP 本 身 并 没有 内 置 FTP 上 传 扩 展 ， 但 PHP WET ftp put 函数 ， 用 于 实现 FTP 
文件 上 传 。 这 里 将 使 用 第 三 方 扩展 实现 FTP 操作 ， 下 载 地 址 为 http://beauty-soft.net/book/ 
php _mvc/vendor/ftp.html - 

下 载 后 将 Ftp.class.php 文件 复制 到 home/Lib/ORG 项 目 类 库 中 。 完 成 后 直接 使 用 import 
函数 村 入 该 类 库 即 可 。 下 和 面 遂 过 代码 渤 示 FTP 文件 上 传 操作 ， 如 以 下 代 人 码 所 示 。 

publie function 1noex()1! 

mortE ("R ORG. FTO”) 2 
SFto=new Prep“ localnost™, 21, "acmin™, 123456) 7 
Tf (EE up iile (T /Public/Images/ logo- no Opt www EE tmp le 
REES () 7 
exit (" EIKI); 
}else{ 


EE ees 


SE DE EE 


} 


} 
如 上 述 代 码 所 示 ， 读 者 可 以 根据 实际 情况 修改 FTP 连接 及 登录 信息 。 需 要 注意 的 是 相 
应 的 FTP 上 传 用 户 需 要 目录 谈 取 及 与 入 权限 。 
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11.2.4 TXH 


在 PHP 开发 中 ， 下 载 文 件 可 以 使 用 curl, fsockopen, file get contents F PA ži. 
ThinkPHP 对 这 些 函 数 进行 了 封装 ， 并 进一步 侧 化 操作 步骤 ， 提 高 开发 效率 。 同 时 还 提供 了 
Header 处 理 ， 能 够 方便 地 对 HTML, TXT 等 文本 进行 下 载 ， 也 能 够 对 JPG, ZIP 等 文件 进行 
下 载 。 下 面 将 结合 示例 代码 介绍 HTTP 扩展 类 库 的 使 用 。 

1. curlDownload 

curlDownload 是 Http 扩展 类 的 一 个 基础 方法 ， 该 方法 基于 CURL 实现 ， 所 以 在 使 用 时 
需要 确保 当前 PH 环境 已 经 开局 CURL 模块 。curlDownload 方法 形式 如 下 。 

static public function curlDownload ($remote,$local) 

其 中 参数 remote 表示 远程 文件 URL 地址， 参数 local 表示 本 地 存放 的 文件 名 ， 市 完整 路 
TZ, WMR local SAHT, MR FRAMAT o 

curlDownload 能 够 方便 地 对 远程 文本 文件 、HIML、XML、 图 片 等 文件 进行 异步 下 载 ， 
如 以 下 代码 所 示 。 


publice function index|) 
1O TE (VORG -NET EE 
HCTOs o curlDownload ("htt / /wu . beauty- soft. net/ index ntm", ™/Pwolic/t ntm") s 


Í 

PHP 内 置 了 file get contents KZG ZAZE PHP JETRA HT FRAR JI 
file_get_contents 并 不 提供 文件 本 地 化 保存 功能 ， 需 要 开发 人 员 手 动 处 理 。 此 外 CURL 无 论 
是 性 能 还 是 灵活 性 都 比 fle get contents KACE Ce, 

2. fsockopenDownload 

fsockopenDownload 是 一 个 强大 的 文件 下 载 方法 ， 访 方法 通 第 用 于 采集 HTML 等 文件 。 
fsockopenDownload 还 提供 了 Cookie 访问 、POST 数据 上 传 、 设 置 下 载 超 时 、 线 程控 制 等 实 
用 功能 。fsockopenDownload 方法 形式 如 下 。 

static public function fsockopenDownload ($url, $conf = array()) 

参数 url 表示 文件 地 址 ; 参数 conf 为 表示 下 载 配置 参数 。conf 配置 项 如 下 。 

> limit: 获取 文本 文件 的 字数 。 

> post: 在 下 载 时 可 以 设置 POST 上 传 数 据 ， 使 用 关联 数组 表示 参数 键 值 ， 如 
array( "user => ”celba )。 
cookie: 在 下 载 时 可 以 携 市 本 地 Cookie 数值 。 
ip: 人 允许 使 用 ip RE url 参数 。 
timeout: 下 载 超时 时 间 ， 以 秒 为 单位 。 
block: FAHER H, SAAN trues 
fsockopenDownload D'DP GES A, UBA PRIBA. 


publice function index () 
import ("ORG.Net.Http"); 
$data=Http: :fsockopenDownload ("http://t.beauty-soft.net/upload/ceiba.jpg",array ( 
u Limie” =>300, 
) ); 
dump ($data); 
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| 

需要 注意 的 是 fsockopenDownload 方法 不 文 持 将 文件 保存 到 本 地 磁 和 机。 如 果 有 此 需要 ， 
读者 可 以 使 用 fopen 处 理 。 

3. download 

download 用 于 直接 输出 文件 保存 对 话 框 ， 方 便 用 户 下 载 文件 。download 能 够 对 常见 的 
TXT、HTML、XML、 图 片 等 文件 提供 下 载 功能 ， 也 能 够 对 压 绚 包 、 二 进 制 文件 等 提供 下 载 
功能 。 议 方法 表现 形式 如 下 。 

static public function download ($filename, $showname='',$content='',$expire=180) 

其 中 参数 filename 表示 存放 于 网 站 的 文件 ， 需 要 市 完整 路 径 ， 参 数 showname 用 于 提示 
用 户 保存 的 文件 名 ; 参数 content 表示 下 载 内 容 ; 参数 expire 表示 该 文件 的 浏览 器 缓存 时 
闻 。 使 用 方式 如 以 下 代码 所 示 。 


publice function index () 4 
STEE (VORG -NGE a PECON) A 
Http: download (mn. ZPublie/t, himi”, a aoa tr. btmlimi: 


运行 效果 如 图 11-2 所 示 。 


正在 打开 静态 文件 (t.html) 
您 选择 了 打开 : 
wg 静态 文件 (t.html) 


为 : Firefox HTML Document (12 字 节 ) 
IR : http://tp.localhost 


您 想 要 Firefox 如 何 处 理 此 文件 ? 
EE 
© 保存 文件 (5) 


以 后 自动 采用 相同 的 动作 处 理 此 类 文件 。(A) 


图 11-2 download 方法 下 载 文 件 效 果 


11.2.5 Socket 套 接 字 编程 


Socket 编程 是 网 络 开发 重要 的 技术 。 与 传统 的 HTTP HE, Socket 最 大 的 特点 是 无 状 
态 ， 所 以 特别 适合 发 送 邮 件 、 实 时 聊天 、 便 件 系 统 交 互 等 通信 应 用 。PHP 在 网 站 开发 中 有 很 
多 应 用 需要 使 用 Socket， 例 如 SMTP 邮件 发 送 、 绥 存 系 统 、 全 文 搜 索 、 文 件 上 传 下 载 等 。 

PHP AG petk TIRK Socket 编程 功能 ， 但 多 数 都 是 通过 PHP 扩展 模块 实现 的 ， 这 样 
能 够 确保 通信 效率 高 效 及 安全 。 事 实 上 PHP 同样 也 提供 了 编程 接口 ， 允 许 开 发 人 员 直 接 在 
PHP 代码 中 调用 Socket。ThinkPHP 对 Socket 编程 进行 了 封装 ， 使 得 编程 模式 更 加 适合 
MVC 编程 ， 下 和 面 将 络 合 代码 进行 介绍 。 

1. Socket 扩展 类 

Socket 类 是 系统 内 置 的 一 个 扩展 类 ， 用 于 方便 地 实现 Socket 编程 。 该 类 包含 了 4 个 成 员 
D, "Fra, 

> connect: 根据 构造 函数 中 定义 的 连接 信息 连接 Socket 服务 器 。 
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> write: 问 Socket 服务 器 写 入 数据 。 

> read: 获取 Socket 服务 硕 数 据 。 

> disconnect: WJF Socket 服务 器 连接 ， 以 市 省 内 存 资源 。 

2. 使 用 Socket 类 

Socket 编程 与 前 面 介绍 的 FTP 一 样 ， 需 要 绑 定 监听 端口 。S$ocket GES F Ge Die Wir o 
口 ， 由 守护 程序 作出 啊 应 ，PHP 通过 Socket 返回 方法 取得 通信 结果 ， 最 后 关闭 连接 。 

XERA Socket WPm. A STETA, Wika Socket 应 用 ， 笔 者 
编写 了 一 个 Socket 服务 问 监 昕 程序 ， 下 载 地 址 为 http://beauty-soft.net/book/php mvc/downy/ 
socket_cli.html， 解 压 后 将 得 到 Windows 可 执行 程序 ， 监 昕 端口 为 7044， 首 次 使 用 时 需要 开 
放 20441, WME 11-3 所 示 。 成 功 运 行 后 ， 程 序 界面 如 图 11-4 ra: 


ve Windows 安全 警报 (| 


e Windows 防火 墙 已 经 阻止 此 程序 的 部 分 功能 


Windows 防火 墙 已 阻止 所 有 专用 网 络 上 的 ServiceConsole 的 某 些 功能 。 


APRN): 


发 布 者 (P): 


B13 (H): C:\s\serviceconsole. exe 


人 允许 ServiceConsole 在 这 些 网 络 上 通信 : 
专用 网 络 ， 例 如 家 庭 或 工作 网 络 R) 


本 地 的 IP 地 址 太 疡 口 与 为 ，192.168.2.13:7944 E 
FFP AP mEt... 


图 11-4 Socket 监听 程序 运行 效果 


需要 注意 的 是 ， 该 程序 使 用 C 可 .0 编写 的 ，Windows XP 需要 自行 安装 Net Framework 
2.0。Windows 7 或 Windows 8 等 操作 系统 已 经 内 置 .Net Framework， 不 需要 安装 。 接 下 来 
就 可 以 在 PHP 中 发 起 Socket 连接 请 求 ， 完 成 Socket 的 常规 操作 了 ， 例 如 上 传 数 据 、 获 取 
数据 等 。 

(1) 上 传 数 据 

上 传 数 据 使 用 write 方法 ， 设 方法 只 有 一 个 参数 ， 即 上 传 数 据 ， 如 以 下 代码 所 未 。 

public function index()1 


LIDOT (ORG UTLI- SOCkSt™) Z 
Şsocket=new Socket (array ( 
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RE EE Be EA 
"port"=>7044 
II: 
SSOCket >Connect EE 
SSOCket ->write ("nello socket"); 
Şsocket->disconnect(); 


} 
数据 上 传 完成 后 ， 服 务 监控 端 会 实时 显示 上 传 的 数据 ， 如 图 11-5 所 示 。 


本 地 的 ITP 地 : PENSA: 192.168.2.13: 7044 
等 待 PHP ZS Cd SE 


E3 a Gë? SR 


请 输 Alz ZF: 


图 11-5 Socket 通信 


(2) 获取 数据 
同样 可 以 使 用 read 方法 获取 服务 端 回 传 的 数据 ， 由 于 B/S 模式 的 关系 ， 在 PHP 中 获取 
Socket 服务 产 数 据 需 要 用 户主 动 刷新 才能 够 成 功 获取 。 如 以 下 代 但 所 示 。 


public function read(){ 
1moort ("ORG. UE1l] .SOCket")s 
Şsocket=new Socket (array ( 
EE 
"port"=>7044 


) ) 8 

SSOCket EE 

dump ($socket->read (1024)); 
Sssocket->disconnect () ， 


} 
可 以 看 到 ，B/S 应 用 并 不 适合 做 Socket 双 回 实时 通信 的 应 有 用。 事实 上 PHP 开发 Socket 
TEA ma 2 zu db mg D Il DI HI ol, 

表面 的 操作 只 是 演示 Socket 通信 ， 没 有 任何 实际 功能 。 在 真实 生产 环境 中 ， 开 发 人 员 
可 以 在 服务 监控 闹 完 成 实际 功能 (Linux 下 可 以 使 用 PHP 编写 守护 脚本 )。 当 然 ， 现 在 很 多 
E E Socket 来 通信 的 ， 不 需 上 自行 编写 。 例 如 后 面 将 介绍 的 Sphinx、Redis 

， 开 友人 员 只 需要 根据 软件 开发 者 提供 的 Socket 通信 协议 完成 相关 调用 即 可 。 

除了 可 以 使 用 PHP 进行 Socket 通信 外 ， 当 前 流行 的 HIML 5 也 将 Socket 通信 定义 为 其 
中 标准 ， 并 命名 为 Web Socket。Web Socket H PHP Socket 拥有 更 丰富 的 特性 ， 感 兴趣 的 读 
者 可 以 参阅 相关 HTML 5 资料 。 


11.2.6 ”定位 当前 位 置 
ThinkPHP 内 置 了 IpLocation 扩展 ， 利 用 该 扩展 能 够 实现 用 户 位 置 定 位 。 这 里 的 位 置 定 
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位 并 非 GPS 高 精度 定位 ， 而 是 利用 访问 用 户 的 ， 得 到 用 户 所 在 的 地 理 位 置 。 所 以 读者 还 
需要 下 载 一 个 地 址 数据 库 ， 卫 地 址 越 多 ， 定 位 的 数据 越 准 确 。 接 下 来 将 结合 示例 代码 ， 介 绍 
IpLocation 扩展 的 使 用 。 


publie function location ()1 
EE (“ORG NeT, Iplhocation™) p 
Şiplocal=new IpLocation(); 
Sdata=$iplocal=>getlocation (112, 94.,255:150™") 5 
dump- data); 


} 

如 上 述 代码 所 示 ，IpLocation 在 构造 图 数 时 共 文 持 一 个 传 参 ， 即 IP ARE, AAN 
UTFWry.dat。getlocation 方法 用 于 取得 当前 用 户 IP 对 应 的 地 理 位 置 数 据 ， 为 了 方便 演示 ， 这 
里 传 入 一 个 互联 网 PPP， 在 实际 应 用 中 ， 不 需要 传 入 参数 ， 系 统 会 自动 得 到 当前 访问 用 户 的 
IP 地 址 。 配 置 完 成 后 ， 还 需要 下 载 UTFWry.dat 数据 库 文 件 ， 下 载 地 址 为 http://beauty- 
soft.net/book/php_mvc/down/utfwry.html - 

下 载 完 成 后 ， 将 文件 解压 缩 ， 并 将 UTFWry.dat 文件 复制 到 ThinkPHP/Extend/Library/ 
ORG/Net 扩展 目录 ， 完 成 上 面 的 步骤 后 ， 现 在 通过 浏览 右 束 可 以 获取 到 用 户 的 地 理 位 置 
言 息 了 。 


11.2.7 Emir 


电子 邮件 发 送 是 网 站 开发 中 重要 的 功能 ， 和 典型 的 应 用 场景 有 会 员 注 册 、 邮 件 订 阅 、 密 码 
找 回 等 。 可 靠 的 邮件 发 送 系 统 能 够 有 效 改 善 用 户 体 验 。PHP AAAS mal mär, Gm är 
在 处 理 邮 件 上 有 具有 人 快速、 稳定 的 特性 ， 所 以 许多 大 型 网 站 都 在 使 用 mal 函数 构建 邮件 系 
统 。 但 是 mail 函数 对 运行 环境 要 求 比较 奇 刻 ， 而 且 不 文 持 SMTP 登录 验证 ， 所 以 很 多 开发 
者 更 加 喜欢 使 用 基于 安全 验证 的 大 型 邮件 系统 “例如 Postfix、SendMail 等 ) 来 构建 PHP 邮 
件 发 送 程序 。 

事实 上 ， 主 流 的 邮件 系统 都 是 基于 SMTP 验证 的 ， 并 用 文 持 Socket 连接 。 所 以 开 
发 人 员 可 以 直接 在 PHP 中 使 用 fsockopen 函数 问 服 务 器 提交 数据 ，SMTP 默认 情况 下 会 
监听 25 号 通信 端口 ， 如 采 校 验 通 过 ， 邮 件 系统 会 直接 执行 Socket 友 送 的 命令 ， 完 成 邮 
Irak. 

ThinkPHP 本 喘 没 有 内 置 邮件 发 送 扩展 ， 接 下 来 将 使 用 第 三 方 扩展 类 库 实 现 邮 件 发 送 。 
下 载 地 址 为 http://www.beauty-soft.net/book/php mvc/vendorsend message.html。 

下 载 完 成 并 解压 后 ， 得 到 SendMessage.class.php 类 库 文 件 ， 将 该 文件 复制 到 项 目 
Lib/ORG 目录 中 。 接 下 来 残 可 以 在 项 目 中 使 用 了 。 如 以 下 代码 所 示 。 


public function sendmail(){ 这 个 段 代码 运行 有 漏洞 ， 需 要 符 换 过 
import ("@.ORG.SendMessage"); 


Şconf=array ( 
Hoort =>25,; 
Uhost =>" lLocalnost" , 
"user"=>"admin", 
"pass"=>"admin", 
E EE 


"mail"=>array ( 
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mail cemol"=>"mail atmi, 
E 
I 
SendMessage::init($conf); 
SEN e a a ncn "mm om 
n eea e a a ee o a ea roe)S 


iE ($infol["stacus”]) < 
$this->success ("邮件 已 经 发 送 成 功 ， 请 登录 邮箱 查看 ")， 
}else{ 


Semis error RE EI, Ti 
} 
} 


需要 注意 的 是 ， 这 里 只 是 完成 了 PHP 代码 方面 的 编程 ， 读 者 在 学 习 或 开发 中 需要 正确 
发 送 邮 件 ， 还 需要 在 本 机 或 远程 服务 器 上 配置 SMTP 邮件 发 送 系 统 〈 建 议 使 用 Postfix)， 关 
于 邮件 系统 的 搭建 ， 读 者 可 以 参阅 笔者 在 博客 里 号 的 一 篇 文 草 ， 网 址 为 http:/beauty- 
soft.net/blog/ceiba/PHP/postfix.html， 在 此 不 再 细 述 。 

SendMessage 扩展 类 库 需 要 在 初始 化 时 配置 数据 选项 ， 也 可 以 直接 在 项 目 配置 文件 中 定 
义 ， 前 者 优先 级 大 于 后 者 。 下 面 将 介绍 名 用 及 重要 的 配置 项 ， 如 表 11-3 PTR 


表 11-3 SendMessage 配置 选项 


全 局 选项 

host 服务 器 IP 地 址 a 
user 登录 用 户 名 空 
pass 登录 用 户 密码 SS 
time ot 登录 超时 时 间 ， 以 秒 为 单位 120 
邮件 配置 

phpmail 是 否 启 用 PHP 内 置 Mail 函数 发 送 邮件 false 
from 发 送 方 电子 邮件 地 址 "e 
auth 是 否 启用 SMTP 验证 true 
relay host 转发 服务 器 卫 ， 为 空 时 与 host 选项 相同 空 

log file 邮件 发 送 记录 日 志文 件 ， 相 对 于 项 目 Runtime/Logs 目录 mail.log 
bodytype 邮件 内 容 类 型 ， 可 选 值 为 HTML 及 TXT HTML 
mail templ 如 果 选 择 HTML 内 容 类 型 ， 使 用 的 邮件 模板 文件 ， 该 文件 相对 于 项 目 Tpl 目录 


其 中 邮件 配置 项 必须 为 mail 元 素 值 。mail templ 配置 项 表示 模板 文件 ， 可 以 指定 HTML 
文件 或 者 PHP 文件 ， 该 文件 支持 如 下 模板 标签 。 

> {title}: HTML 邮件 标题 。 

> {body}: HTML 邮件 正文 内 容 。 

> {form}: 邮件 发 送 者 。 

> {time}: 邮件 发 送 时 间 。 

一 个 最 简单 的 邮件 模板 如 以 下 代码 所 示 。 

E 

<head> 
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<meta htcpo-ecuiy= Content-Type" content=" text/html; Cnarset=uEt = /> 

<link type="text/css" href="http://beauty-soft.net/css/main.css" title="style" 
media="screen" rel="stylesheet" /> 

titlezititlet/z/titlez 

</head> 

<body> 

{body} 

</body> 

</html> 


所 有 配置 项 文 持 在 项 目 配置 文件 中 配置 ， 只 需要 将 表 11-3 的 配置 项 赋值 给 SendMessage 
元 素 即 可 ， 如 以 下 代码 所 示 。 


"SendMessage"=array ( 
WEE 
let EEN 
"user"=>"admin", 
pasg i=>"23456"; 


"mail"=>array ( 


mail cemol"=>"mail - mrmi , 
Erom“ =>" ki R060559- COM" , 


11.3 ”数据 处 理 


通过 扩展 关 库 ， 能 够 对 数据 进行 深度 处 理 ， 例 如 数据 分 页 、 数 据 加 密 、 纺 码 转换 等 。 为 
了 方便 操作 ， 提 高 开发 效率 ， 系 统 还 提供 了 String 字符 冲 处 理 类 库 ， 使 用 该 类 库 能 够 对 字符 串 
进行 截取 支持 中 文 )、 随 机 生成 学 符 串 、 编 码 检 测 每 ， 接 下 来 首先 从 数据 加 密 开始 介绍 。 


11.3.1 ”数据 加 密 


数据 加 密 是 网 站 开发 中 重要 的 安全 技术 ，PHP 共 文 持 两 大 类 加 密 方式 : 一 种 是 不 可 逆 的 
加 密 方 式 ， 即 单 向 加 密 ， 例 如 常见 的 Mis, Crypt, Des 等 ;另外 一 种 是 支持 双向 操作 的 加 
密 方式 ， 例 如 Base64、Xtea 等 。 接 下 来 将 分 别 介绍 系统 内 置 的 和音 用 加 密 处 理 扩 展 。 

1. Des 

Des 是 一 种 支持 加 密 及 解密 的 加 密 技术 ， 在 Linux, UNIX 等 操作 系统 底层 实现 中 ， 已 经 
在 大 量 应 用 。PHP Har Des 加 密 ， 使 用 方式 也 比较 简单 。ThinkPHP 内 置 了 Des 扩展 
KE, EHZ EKI Des 加 蜜 解密 ， 更 加 符合 MVC 编程 模式 ， 如 以 下 代码 所 示 。 

EE EE ae 


(et GE eh Ee BEE EE EE ) 
STEE Ee Des") p 


5des=new Des () ; 

WAS 

$data=$des->encrypt ("123456789", "mykey"); 
echo Saatas 

/ /解密 

echo $ces=>cecrypt (Sdata, "mykey™) s 
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在 实际 应 用 开发 中 ， 读 者 可 以 使 用 数据 库 等 拷 术 将 加 加 后 的 字符 串 保 存 ， 使 用 时 和 直接 解 


密 即 可 还 原 数 据 。 
2. Base64 


Base64 加 窄 方式 是 网 络 上 最 第 用 的 加 密 方 式 ， 典 型 的 应 用 有 URL 传 仁 。Base64 通过 
把 3 个 8 位 bit) 的 字符 转换 为 4 个 6 位 的 字符 ， 最 后 使 用 0 补 够 8 位， 最 终 实现 数据 加 
密 。 解 密 时 也 必须 要 沿 这 个 步骤 反 同 运算 。Base64 类 库 的 使 用 与 Des 类 库 相 类 似 ， 如 以 下 
代码 所 示 。 


publice function index|) i 
header ("Content-Type text/html; charset=uti-=8" E 
import ("ORG.Crypt.Base64"); 
Sbase64=new Base64 (); 
/ /加密 
şdata=Ş$base64->encrypt ("123456789abcdefglmn", "mykey"); 
echo $data; 
/ /解密 
echo sbase64->decrypt ($data, "mykey"); 
} 


3. Rsa 

Rsa 是 由 Ron Rivest, Adi Shamirh JL mb GH lU Dr, ME 
己 经 成 为 一 种 国际 化 通用 标准 的 加 密 方式 。Rsa 使 用 的 是 数值 相 乘 加 密 方 式 ， 能 够 实现 对 超 
长 学 符 串 的 加 密 。Rsa 是 一 种 高 效 的 数学 加 密 方式 ， 由 于 位 数 高 ， 能 够 有 效 防 止 密码 猜测 等 
攻击 。Rsa 常用 于 数字 证 书 加 密 、 便 件 级 的 蕊 厂 加 密 每 。 PHP 对 Rsa 提供 了 国际 化 文 持 ， 如 
以 下 代码 所 示 。 


publice function index|) 
header ("Content=Types text/html; Charset=utt=8" ) 7 
LDOCE ("ORG Crypt- Rsa") g 


Srsa=new Rsa(); 
/ /加密 
odata- rsa EE ee EE 
echo $data; 
/ /解密 
echo Srsa=>decryot (Sadata, "mykey™) s 
l 


4. Crypt 

在 PHP 中 MD5 等 加 密 方 式 都 可 称 为 Crypt， 但 这 里 的 Crypt 并 不 是 指 MD5， 而 是 
ThinkHPP 内 置 的 一 种 实用 的 双向 加 密 类 库 ，Crypt 类 库 能 够 对 包括 中 文 在 内 的 常见 字符 进行 
加 密 。Crypt 关 库 的 加 密 方式 非常 简单 ， 根 据 字 符 串 的 长 度 进行 循环 截取 。 此 外 Crypt XE 
支持 Besa64 加 密 。 如 以 下 代码 所 示 。 


public function incex() q 
header ("Content-Type text/html; charset=utf=8" ) s 
maoOrE ("ORG - Crypt- Crypt) z 


SCrypt=new Crypt); 

Ea 
3 
echo $data; 

echo "<br/>", 

IA 
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echo $Crypt->decrypt ($data, true); 

} 

encrypt 方法 与 decrypt 方法 都 文 持 两 个 参数 。 其 中 人 参数 1 表示 加 密 的 字符 串 ， 参 数 2 表 
示 是 否 使 用 Base64 位 加 密 ， 默 认为 false。 

5. Hmac 

Hmac 类 库 是 一 个 哈 希 运算 加 密 类 库 ， 包 含 了 shal 与 md5 方法 。 前 者 用 于 实现 SHAI 
加 密 ， 后 者 用 于 实现 Mis 加 密 ， 这 两 种 加 蜜 方式 都 是 单 癌 不 可 逆 的 ， 非 党 适用 于 加 密 用 户 
Si, kim GI, Hmac 类 库 的 使 用 。 


publie functiom index () q 


moort (ORG, Crypt- Emac™) p 


ES 
echo "<br/>"; 
Echo CNDS me mde o ke a eo 


} 


11.3.2 ”数据 编码 转换 


ThinkPHP 创建 网 站 时 使 用 UTF8 编码 ， 包 括 数 据 库 操作 默认 情况 下 也 使 用 与 文件 系统 
相同 的 编码 。 但 有 些 第 三 方 应 用 ， 使 用 的 并 非 UTF8 编码 ， 如 果 在 当前 应 用 中 调用 ， 将 会 出 
现 异 党 ， 例 如 乱码 、 解 释 错 误 等 。 事 实 上 PHP 已 经 内 置 了 iconv 函数 用 于 处理 字符 编码 ， 但 
只 能 处 理 字符 串 。ThinkPHP 内 置 了 CodeSwitch 类 库 ， 能 够 方便 地 对 PHP 及 HTML 文件 进 
行 转 何 ， 文 持 批 量 转 码 。 转 但 后 开 友 人 员 残 可 以 方便 地 调用 不 同 编 但 的 关 库 及 冰 数 。 

1. DetectAndSwitch 方法 

DetectAndSwitch 方法 可 以 方便 地 对 单个 PHP, HTML, JS 等 文件 进行 编码 转换 ， 
DetectAndSwitch 方法 形式 如 下 。 

static function DetectAndSwitch ($filename, $out charset) 

其 中 参数 filename KIR RRAN: AŽ out charset 表示 转换 后 的 编码 ， 接 受 
的 输出 编码 与 icony 函数 一 样 。 如 以 下 代码 所 示 。 

oe 


SEET 
WE EB EE deif DE ER E Es 


dr MET DI KE e RRR D, (Cl E dt Linux 系统 下 ，Web IRI REX Z 
文件 必须 具备 可 讯 可 写 权 限 。 

2. CodingSwitch 方法 

很 多 第 三 方 尖 库 都 不 是 由 单个 文件 构成 的 ， 而 是 由 一 系列 的 类 库 构 成 的 。CodingSwitch 
方法 可 以 实现 对 整个 目录 下 的 PHP 或 者 HTML 等 文件 进行 编码 转换 ， 函 数 形式 如 下 。 


static function CodingSwitch($app = "./",Şcharset='UTF-8',$mode = "FILES",S$file 


types = array(".html",".php")) 

其 中 参数 app 表示 日 录 路 径 ;， 参数 charset 表示 文件 转换 的 目标 编码 ;参数 mode 表示 日 
录 过 历 模式 ， 保 持 默认 即 可 ; 参数 file types 表示 文件 匹配 后 级 名 ， 只 有 [匹配 的 文件 才 进 行 
FA. CodingSwitch 方法 的 使 用 非常 徐 单 ， 如 以 下 代码 所 示 。 


Buolie funetion imoex()! 
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LImMPOrT (ORG Ee Ee Ger 
enee Eeer ee Dee EE Ee Kee kee EE 


11.3.3 Seng 


在 前 面 的 CURD 操作 中 ， 己 经 简单 涉及 数据 分 页 内 容 。 接 下 来 将 详细 介绍 ThinkPHP 内 
GI Page 基础 数据 分 页 类 。Page 关 库 是 一 个 扩展 类， 使 用 时 需要 额外 引入 。 一 个 最 简单 的 
数据 分 页 功能 ， 人 代码 如 下 所 未 。 


publie function lncoex()! 
$article = M("Article™) g 
LMDOrT (ORG UTLI. Page") z 


SCount = Bartiele-zeount l); 
$Page = new Page ($count, 3); 
Sshow = $Page->show (); 


$list = Sarticle->l11imit ($Page=>fLESTROW.", " -7 ŞPage=>listRows)=>selecr |); 
Ş$tchLs=>assign(" list"; $list)s 

Stis >assign (pagel snow) 

ER EE BEE Eier 


} 

对 应 的 视图 模板 文件 代码 如 下 所 示 。 

</style> 

<div class="main"> 

<ul> 

<volist nemes“ list™ osiyo" > 
<li><!--{$vo.title}--></li> 

</volist> 

</ul> 

<ul class="page"> 

<!--{$page}--> 

</ul> 

</ ALYS 


上 上述 视 图 模板 代码 应 用 了 Layout 全 局 布局 ， 其 中 变量 page 即 为 接 下 来 需要 重点 介绍 的 
分 页 变量 。 效 果 如 网 11-6 所 示 。 


ZO 


Giel | 所 有 日 志 | IS: 


谊 中 心 | 点 表 日 志 


库 克 训 苹 采 地 图 缺陷 向 用 户 道 次 : 


未 提供 一 流体 验 


D 
索尼 投资 6.44 亿 美元 成 为 奥 林 巴 斯 第 一 大 股东 E - 
夏 莹 称 会 制造 足够 iPhone5 所 需 的 显示 屏 

网 站 公告 
6 条 记录 1/2 页 下 -页 12 我 的 博客 开通 了 ， 欢 迎 来 


首页 | 联系 方式 | 关于 本 站 | 全 文 搜索 | 订阅 


Copyright by me and not one other person 2012 


图 11-6 ”数据 分 页 效 末 
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Page 对 外 公开 了 一 些 成 员 属 性 ， 在 调用 show 方法 前 ， 开 发 人 员 可 以 通过 改变 成 员 属 性 
的 值 ， 直 接 改变 分 页 的 形式 ， 例 如 分 页 数量 、 外 观 等 。 公 开 可 调用 的 成 员 属 性 如 表 11-4 所 


表 11-4 Page 类 成 员 


默 认 值 
listRows 分 页 导航 栏 显示 的 分 页 数量 《序列 数 ) 5 
parameter 分 页 数 跳 转 时 要 带 的 参数 Ss 
listRows 每 页 的 数据 量 25 
firstRow 数据 查询 的 起 始 行 0 
setConfig 成 员 方 法 ， 用 于 改变 分 页 导航 栏 的 文本 信息 z 


setConfig 是 一 个 成 员 方 法 ， 用 于 改变 分 页 导航 栏 的 文本 信息 ， 例 如 将 “下 一 页 ” 改 为 
“Next” 可 通过 该 方法 进行 设置 。setConfig 方法 形式 如 下 。 

function setConfig ($name, $value) 

其 中 参数 name 表示 分 页 导航 栏 项 ， 参 数 value 表示 相应 的 值 。 分 页 导航 栏 可 设置 的 选 
项 如 下 : 

> header: 分 页 导航 栏 的 头 部 信息 ， 默 认为 “条 记录 ” 

> prev: 返回 上 一 页 的 显示 文本 ， 默 认为 “上 一 页 ” 

> next: 前 往 下 一 页 的 显示 文本 ， 默 认为 “下 一 页 ” 

> first: 开始 页 显示 文本 ， 默 认为 “第 一 页 ”。 

> last: 最 后 一 页 显示 文本 ， 默 认为 “最 后 一 页 ”。 

> theme: 分 页 导航 栏 的 外 观 主题 ， 改 变 该 值 可 以 直接 改变 分 页 导航 栏 的 文本 显示 格式 


11.3.4 ”日 期 数据 


在 网 站 开发 中 处 理 时 间 日 期 数据 是 比较 烦琐 的 工作 ， 例 如 计划 日 期 时 兰 、 日 期 转 星 期 、 
指定 日 期 倒计时 等 。 系 统 内 置 了 Date 扩展 类 ， 能 够 让 日 期 数据 处 理 变 得 快捷 、 人 简单 。 接 下 
来 将 介绍 Date 类 第 用 的 方法 。 

1. dateDiff (比较 日 期 跨度 )》 

dateDiff 成 员 方 法 用 于 比较 两 个 日 期 数据 间隔 的 时 间 。 返 回 的 结果 以 “年 人 “月 和 
日 和 “时 入 “分 和 “ 秒 ” 作 为 计算 结果 。dateDi 人 方法 形式 如 下 。 

function dateDiff ($date, Selaps = "d") 

其 中 参数 date 表示 与 构造 函数 日 期 比较 的 日 期 数据 ;参数 elaps 表示 比较 跨度 。elaps 文 持 6 
个 参数 值 ， 用 于 返回 结果 值 的 计算 单位 ， 分 别 为 y ER M(H) w (星期 )、h (小 时 )、m 
(分 钟 )、s( 秒 )。dateDiff 方法 在 运算 时 与 构造 冰 数 参数 进行 比较 ， 如 以 下 代码 所 示 。 

publice function index|) i 

import (ORG. Util Date”); 


Sdatenew Date(2012-06-131); 
dump ($date->dateDiff ("2013=1L11=15™, ma) ) ; 
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上 述 代 码 运行 结果 为 “1.4246575342466”， 即 1.4 年 。 

2. timeDiff (以 中 文 显示 日 期 跨度 ) 

timeDIiff 方法 与 dateDiff 类 似 ， 不 同 的 是 timeDiff 并 非 以 浮 点 数值 为 返回 结果 ， 而 是 
回 友好 的 中 文 格式 。timeDi 企 方法 形式 如 下 。 

timeDiff( Ştime ,Şprecision=false) 

其 中 参数 time 表示 与 构造 函数 日 期 比较 的 日 期 数据 ;参数 precision 表示 结果 精度 。 
precision 参数 值 与 表面 介绍 的 elaps 参数 值 一 样 。timeDiff 的 使 用 如 以 下 代码 所 示 。 

public function index|) i 

IMPTE (ORG UCLL. Date”) z 


sdate=new Date ("2012-06-13"); 
dump ($date->timeDiff ("2013-11-715")); 


e 


上 述 代 码 运行 结果 为 “1 年 前 ”使 用 timeDiff 方 法， 可 以 方便 地 模拟 微 博 、 日 志 之 类 的 
日 期 显示 应 用 。 

3. yearToCh 《年份 转 中 文 ) 

一 些 中 文 应 用 通常 需要 将 数字 转换 为 中 文学 从 串 。yearToCh 方法 能 够 将 利 见 的 数字 年 份 
转换 为 中 文 格式 的 年 份 ， 并 且 文 持 公元 元 年 的 转换 。yearToCh 方法 形式 如 下 。 

function yearToch( $yearStr ,$flag=false ) 

其 中 参数 yearStr 表示 与 构造 函数 日 期 比较 的 日 期 数据 ;参数 flag 表示 是 否 显 示 公 元 元 
年 。yearToCh 方法 的 使 用 如 以 下 代码 所 示 。 


publie function index|) i 
MDOT ("ORG. UTLI. Dace")s 
Sdate=new Date ("2012-06-13"); 
dump (Scate >yearioCn ("2012", true) ) 5 


} 

上 述 代 但 运行 结果 为 “公元 二 零 一 二 ”。yearTocCh 方法 不 仅 可 以 转换 年 份 ， 还 可 以 对 普 
通 的 数字 进行 转换 。 

4. magiclnfo (计算 生 肖 ) 

magicInfo 方法 用 于 运算 构造 函数 指定 日 期 所 属 的 生肖 、 星 座 等 信息 。magicInfo 方法 形 
AA Fe 

function magicInfo ($type) 

其 中 参数 type 表示 返回 的 结果 类 型 ， 共 文 持 3 种 类 型 ， 分 别 如 下 : 

> XZ: 返回 星座 。 

> G2: BEF Dm. As As Ta Rs GP. Tt, Së. H. e, D. 

E DH... mm. BÊ, H. ZZA. 
> SX: 返回 十 二 生肖 。 
magicInfo 方法 的 使 用 如 以 下 代码 所 示 。 


EE function index|) i 
Ort (ORG. Util. Date") z 
Sdate=new Date ("2012-06-13"); 
dump (Ş$date->magicInfo("XZ")); 


} 
上 上 述 代 三 返回 结果 为 “双子 座 ” 读者 可 以 尝试 更 改 type 参数 值 ， 观 察 magicInfo 运算 
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结果 ， 如 果 需 要 多 语言 ， 可 参考 第 9 章 9.3.3 WRI. 


11.3.5 Input 类 


PHP 是 一 种 弱 类 型 开 友 语言 ， 对 字符 串 变 量 能 够 日 动 识别 ， 在 处 理学 符 串 上 一 下 是 强 
项 。 在 传统 的 PH 开发 中 ， 用 于 处 理 字 符 串 的 技术 的 有 函数 、 正 则 以 及 扩展 等 。Input Zb 
是 ThinkPHP 提供 的 一 个 字符 串 输 入 处 理 扩 展 ， 能 够 方便 地 对 字符 串 进 行 安全 过 滤 、 转 义 
等 ， 下 面 将 对 Input 扩展 类 公开 可 调用 的 方法 进行 介绍 。 

1. makeLink (自动 匹配 链接 )》 

在 网 站 开发 中 ， 经 党 需要 对 页 和 面 中 的 网 址 学 符 串 进行 目 动 匹配 。 例 如 用 户 在 发 表 微 博 
时 ， 只 需要 输入 文本 网 址 ， 系 统 会 目 动 将 该 网 址 转换 为 标准 的 HTML 链接 代码 。 这 对 用 户 
而 言 ， 提 高 了 用 户 体验 ， 对 于 网 站 而 言 ， 提 高 了 安全 性 《用户 只 能 输入 文本 )。 在 传统 的 
PHP 中 ， 实 现 上 述 功 能 可 以 使 用 正则 蕉 换 ， 在 MVC 编程 中 ， 这 些 步 又 简化 为 使 用 类 库 方 
法 。makeLink 束 是 一 个 基于 正则 蕉 换 实 现 的 URL 处 理 方法 ， 形 式 如 下 。 

static public function makeLink ($string) 


参数 string 表示 URL 字符 如 文本 《需要 带 http)， 使 用 方式 如 下 。 


public function incex () q 
LIMO rEt (ORG UTLI. IMPuT™) z 
EE // beauty- ofc. net" s 
dump (Input: :makeLink ($str) );，; 


| 

URL 协议 文 持 https, ftp 等 ， 运 行 结果 为 “<a href="http://beauty-soft.net" title="http:// 
beauty-soft.net" rel="external">http://beauty-soft.net</a>” o 

2. truncate 〈 自 动 省 略 字符 ) 

truncate 方法 能 够 对 字符 串 进 行 目 动 省 略 ， 被 省 略 的 学 符 寺 认 使 用 “.…” 代 和 登 。truncate 
JE mt, 

static public function truncate ($string, $length = '50') 

其 中 参数 string 表示 需要 进行 省 略 处 理 的 字符 串 ; 参数 length 表示 省 略 的 字符 数 。 使 用 
方式 如 以 下 代码 所 示 。 

public function index (){ 

import ("ORG.Util. Input"); 


Sat rei DH DM SLIT, 
cumo (Input: s truncate (Sst; 3) ); 


} 

上 述 代 码 的 运行 结果 为 “了 P.… VC 实战 ”。 

3. safeHtml 安全 HTML) 

safeHtml 能 够 对 HTML 中 不 安全 脚本 进行 过 泪 ， 例 如 JavaScript、 转 义 字 符 等 。 
safeHtml 通常 用 于 过 滤 表 单 输入 ， 防 止 用 户 输入 HTML 代码 破坏 数据 安全 性 。 也 可 以 用 于 调 
用 第 三 方 页 面 代码 时 过 滤 其 中 的 危险 代码 。safeHtml 方法 如 下 。 

static function undoHsc ($text) 

参数 text 表示 需要 安全 过 小 的 HTML 代码， 可 以 来 日 于 表单 或 者 远程 HTML。 使 用 方 
CT 
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publie function incex() q 
LMPDOrT ("ORG UTLI. Input™) z 


shml=" 
<hml> 
<head- title Se 
<SCELPT> 
EECHER TEST) | 
alere (“atmi ) E 
} 
</scipt> 
</head> 
<body> 
SE a a a 
</body> 


<html> 
echo Inputs: :safeatml (Sami) 5 


} 

代 人 运行 结果 为 “<hml> 测试 function test) { alert(html); } </scipt> 测试 </a>”。 与 之 相 
类 似 的 方法 还 有 deleteHtmlTags， 访 方法 不 仅 过 小 危险 的 HIML 代码 ， 还 对 HTML 代码 进行 
过 小 ， 只 留 下 纯 文 本 数据 。 

d. nl2Br 自动 换行 )》 

nl2Br 方法 能 够 对 用 户 输 入 的 回 车 符 转 换 为 HTML 代码 中 的 <br 放 换行 从 ， 方 法 形式 
ul he 

static public function nl2Br ($string) 


参数 string KIR mi e A RITARA. BEHA AU PART 


public function incex() q 
1m2ort (ORG UTLI. IMPuT™) 
EEC 
SKAR"; 
dump (Input: snl2Br (Ssetr)); 


} 
上 述 代 码 的 运行 结果 为 “PHP MVC<br 六 实战 <br />”. 


11.3.6 GD 库 绘 制图 形 


GD 图 形 人 处理 库 (简称 GD JE) 是 基于 Zend Engine 的 强大 基础 类 库 ， 了 网络 上 许多 数据 
矢量 图 都 是 由 GD 库 绘 制 的 ， 例 如 网 站 数据 统计 图 、 股 市 行情 分 析 图 、 员 工 考 勤 图 等 。 当 然 
这 只 是 GD 库 的 基础 功能 之 一 ， 事 实 上 GD 库 能 够 处 理 与 图 片 有 关 的 多 数 操作 ， 例 如 前 面 介 
绍 的 图 形 验证 码 、 图 上 请 水 印 等 都 是 基于 GD 库 实现 的 。 接 下 来 将 继续 使 用 GD 库 绘 制 数据 统 
计 图 。 

要 实现 绘制 图 形 ， 首 先 需 要 确认 当前 PH 环境 是 否 已 安 GD 库 ， 可 以 在 phpinfo HA 
看 。 在 确认 文 持 后 ， 束 可 以 直接 使 用 PHP 函数 调用 相应 的 接口 了 。 

与 GD 库 操 作 相 关 的 函数 非常 多 ， 这 里 为 了 方便 演示 ， 将 使 用 第 三 方 扩 展 类 库 
DrawGraph "CN äre RI ml, CS nl DL o 4E http://beauty-soft.net/book/php_mve/vendor/ 
draweraph.html 网 址 下 载 。 

解压 后 得 到 DrawGraph.class.php 扩展 类 库 文 件 ， 将 该 文件 复制 到 项 目 Lib/ORG Ha 
SIN 
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中 ， 接 下 来 就 可 以 在 项 目 中 导入 该 类 库 了 ， 如 以 下 代码 所 未 。 


public function 1inoex()1! 
import ("@.ORG.DrawGraph"); 
Şdata=array ( 

EEN 
"name"=>"1 月 份 "， 
"max"=>10, 

E 

"1 "=>array ( 
"name"=>"2 HH", 
"max"=>90, 

E 

ENEE h 
"name"=>"3 月 份 "， 
"max"=>80, 

e 

"3"=>array ( 
"name"=>"4 HY", 
"max"=>40, 

), 

"4"=>array ( 
"name"=>"5 月 份 "， 
"max"=>400, 

E 

He 
"name"=>"6 HW", 
"max"=>800, 

II: 


Şconf=array ( 
aa 
e 
E 
JI: 
DrawGraphs s EE Ee EE E 
EE eeler 


如 上 述 代码 所 和 示 ， 变 量 data 表示 统计 数据 ，DrawGraph 类 库 在 生成 图 形 时 ， 需 要 根据 统 
计 结 果 进 行 绘制 ， 最 终 效果 如 图 11-7 所 示 。 


1 月 份 "Hm 3 月 份 4 月 份 5 月 份 “月份 X 
10 元 ”90 元 8 元 40% 元 400 元 8007 


图 11-7 GD 库 绘 制 数据 图 形 效果 
改变 参数 conf 的 值 将 直接 决定 绘图 的 外 观 效 果 ， 其 中 width, font, filename 选项 为 必 选 
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项 。 如 表 11-5 所 示 。 


SS 11-5 DrawGraph 配置 项 


LL mg 
width 画布 宽度 Ge 
type 图 片 类 型 png 
greenColor 画布 背景 颜色 255,255,255 (白色 ) 
lineColor 网 络 线条 颜色 0,0,0 (黑色 ) 
pillarColor ÆRE 255,0,0 (红色 ) 
XYColor XY 轴 颜 色 0,0,255 CHE) 
font 字体 文件 Runtime/simhei.ttf 
unit 数据 数值 单位 SR 
headout 是 否 在 浏览 器 直接 输出 true 
filename 图 片 文 件 名 称 Së 


其 中 width 表示 男 布 的 宽度 ， 如 果 为 空 系 统 则 根据 数据 的 数量 进行 智能 计算 ;参数 font 
表示 字库 ， 该 文件 可 以 在 操作 系统 目录 中 获取 ; 参数 headout 表示 是 否 在 浏览 项 中 直接 输出 
图 片 ， 如 果 和 需要 保存 图 片 ， 可 以 设置 为 false; 参数 filename 表示 图 片 名 称 ， 如 果 headout ig 
置 为 false 时 ， 访 参数 不 能 为 衬 〈 市 体 存 路 径 ， 例 如 /PublicImages/t.png )。 

需要 注意 的 是 ， 如 果 headout 为 true 时 ， 需 要 确保 输出 图 片 前 不 能 有 任何 数据 被 输出 ; 
当 headout 设置 为 false 时 ， 般 要 确保 存放 目 杂 可 读 可 写 。 


11.4 行为 扩展 


行为 是 一 项 非常 灵活 及 实用 的 功能 ， 系 统 内 置 的 很 多 功能 都 是 基于 行为 实现 的 ， 例 如 
URL 路 由 、 多 语言 、 绥 存 生 成 等 。 同 时 系统 提供 了 一 系列 内 置 行为 扩展 ， 使 用 这 些 扩展 ， 开 
发 人 员 可 以 更 加 方便 管理 项 目 执行 流程 。Behavior 类 是 行为 类 库 的 基 类 ， 系 统 允 许 开 发 人 员 
在 继承 该 闫 的 情况 下 创建 目 定义 行为 扩展 。 接 下 来 将 首先 理解 行为 概 您 ， 然 后 再 分 别 介 绍 行 
为 的 实际 应 用 。 


11.4.1 行为 概述 


传统 的 PHP 开 友 通常 都 是 根据 URL 请 求 然后 进行 后 台 运 算 的 。 在 这 一 过 程 中 ， 开 发 人 
员 需 要 对 URL 请 求 进行 处 理 ， 例 如 安全 检测 、URL 映射 等 ， 只 能 在 运算 进行 前 使 用 特定 的 
文件 进行 拦截 或 处 理 。 在 项 目 小 的 情况 下 这 不 会 成 为 问题 ， 但 当 项 目 文件 太 多 时 ， 拦 规 的 效 
率 永 明显 不 足 ， 此 时 再 增加 搓 惟 功 能 就 会 变 得 困难 。ThinkPHP 的 行为 机 制 本 质 上 是 一 种 
URL 搓 帘 处 理 机 制 ， 行 为 的 引入 把 过 去 没有 生命 周期 概念 的 PHP 应 用 添加 了 生命 周期 的 概 
念 。 行 为 机 制 将 项 目 拆 成 略 干 个 功能 片段 ， 在 应 用 执行 的 不 同 阶段 ， 这 些 功 能 片段 会 被 执 
行 ， 同 时 执行 与 之 相对 应 的 行为 。 

系统 内 置 了 9 大 行为 ， 分 别 为 CheckRoute 、LocationTemplate 、 ParseTemplate 、 


320 O O 


第 11 章 ThinkPHP 扩展 


ShowPageTrace、ShowRuntime、TokenBuild、ReadHtmlCache、WriteHtmlCache 。 这 些 行为 
在 程序 运行 到 不 同 的 阶段 时 会 被 触发 (可 以 同时 触发 多 个 行为 )， 如 图 11-8 所 示 。 


app_init 《初始 化 程序 ) 执行 ReadHtmlCache 行 为 
route check (Wé 执行 CheckRoute 行 为 
app_begin 《开始 运行 应 用 ) 


action_name 《定位 控制 器 ) 


action_begin (i 调用 控制 器 ) 


view_begin 〈 调 用 视图 引擎 ) 
TE 
(Index.php) 
view_template 〈 载 入 模板 ) 
view Template 〈 解 释 视 图 ) 


执行 CheckRoute 行 为 


view_parse 《解释 标签 》 执行 ParseTemplate 行 为 
执行 ShowRuntime、 

view_filter GIERE ) TokenBuild、 
WriteHtmlCache 行 为 


view_end (视图 解释 结束 ) 执行 ShowPageTrace 行 为 


action end 《控制 络 调用 结束 ) 执行 ReadHtmlCache 行 为 


app end 程序 结束 ) 


图 11-8 ThinkPHP 行为 执行 过 程 

内 置 行为 是 确保 系统 能 够 运行 的 基础 ， 一 般 情 况 下 开发 人 员 不 需要 扩展 内 置 的 行 
为 。Behavior 类 是 所 有 行为 类 库 的 基 类 ， 系 统 允 许 开 发 人 员 直 接 继承 该 类 实现 行为 扩展 。 如 
图 11-8 所 示 ， 项 目 首先 执行 的 行为 标签 是 app_init， 一 直到 app end 标签 ， 中 间 的 标签 
有 些 有 对 应 的 行为 ， 有 些 没 有 。 目 定义 行为 扩展 通 弟 了 吏 是 定义 在 这 些 只 有 标签 没有 行为 
的 运行 阶段 中 。 

这 里 有 一 个 非常 重要 的 标签 概 仿 。 所 谓 的 标签 事实 上 就 是 行为 的 别名 ， 项 目 不 能 直接 执 
行 行为 类 库 ， 只 能 执行 标签 (标签 也 可 以 日 定义 )， 一 个 标签 可 以 对 应 右 干 个 行为 兴 ， 如 果 
家 要 调用 扩展 行为 (包括 系统 内 置 的 扩展 行为 )， 可 以 直接 在 Conf/tags.php 文件 中 定义 标签 
执行 的 行为 类 。 
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标签 允许 同时 出 现 ， 默 认 情 况 下 ， 后 面 出 现 的 标签 会 履 兰 前 面 出 现 的 标签 。 如 采 标 签 为 
宇 ， 则 系统 放弃 执行 ， 例 如 'app end=>array(0)。 接 下 来 将 结合 示例 代码 ， 详 细 介 绍 内 置 行为 
的 使 用 。 
11.4.2 ”内 置 行为 扩展 

表面 介绍 过 ， 系 统 内 置 的 行为 是 系统 正 弟 执行 的 基础 ， 所 以 开 友 人 员 不 需要 改动 这 些 扩 - 
展 。 事 实 上 系统 已 经 提供 了 一 系列 行为 扩展 ， 在 项 目 中 使 用 时 ， 直 接 配置 即 可 〈 上 所 有 内 置 行 
为 扩展 默认 都 是 被 茶 用 的 )。 下 面 分 别 介绍 。 

1. AgentCheck (检测 浏览 器 代理 ) 

AgentCheck 行为 是 一 个 系统 提供 的 扩展 行为 ， 用 于 检测 当前 访问 用 户 是 否 使 用 代理 访 
问 ， 如 条 是 则 蔡 止 访问 。 开 局 该 行为 只 需要 两 步 。 首 先 打 开 项 目 配置 文件 ， 开 局 
LIMIT PROXY VISIT 配置 项 ， 如 以 下 代码 所 示 。 

“LIMIT PROXY VISLT"=>EEUe, 

然后 打开 项 目 Conf/tags.php 文件 (如 果 不 存 在 则 需要 手动 创建 )， 配 置 行为 标签 ， 以 便 
系统 正确 执行 ， 如 以 下 代码 所 示 。 


return array ( 


'app init"=>array ("AgentCheck") ， 


) 7 

如 上 述 代 人 码 所 示 ， 这 里 执行 AgentCheck 行为 的 运行 阶段 为 app_init《〈 即 初始 化 程序 )。 
读者 可 以 根据 图 11-8 所 示 标 识 的 流程 ， 在 相应 的 标签 中 定义 需要 的 扩展 行为 。 通 过 前 和 面 的 
步骤 ， 当 前 项 目下 的 网 页 束 具 备 检测 是 人 否 代理 访问 的 功能 

2. BrowserCheck (防止 浏览 器 刷新 ) 

BrowserCheck 行为 利用 Cookie 防止 用 户 频 索 刷 新 浏览 奏 ， 广 行为 共有 一 项 配置 ， 如 以 
下 代码 所 示 。 

"LIMIT REFLESH TIMES"=>5, 

配置 项 LIMIT REFLESH_TIMES 表示 刷新 的 间隔 ， 以 秒 为 单位 。 设 置 完 成 后 ， 上 只 需要 
在 tags.php 文件 中 定位 运行 标签 即 可 ， 如 以 下 代 但 所 示 。 


return array 人 


'app inirt“=>array ("Browsercheck") , 
E 
3. CheckLang (检测 多 语言 ) 

CheckLang 行为 在 9.3 节 已 经 有 评 细 介绍 ， 在 此 不 再 细 述 。CheckLang 行为 共 文 持 4 个 
配置 项 ， 如 以 下 代码 所 示 。 


TCT = ee // 是 否 开启 多 语言 
NE = // 是 否 目 动 侦 测 浏览 需 语 言 
NO // 允许 切换 的 语言 列表 用 喜 号 分 隔 
RN => 'zh-ch', // 默认 语言 包 


4. CronRun〔 计 划 任 务 ) 

这 里 的 计划 任务 与 Linux 中 的 计划 任务 不 是 一 个 概念 ，CronRun 行为 需要 由 浏览 费解 
发 ， 执 行 的 任务 指令 为 Web 可 执行 脚本 ， 例 如 JavaScript、PHP、HTML 等 。 要 局 动 计划 任 
务 ， 只 需 4 个 步骤 。 首 先 配置 计划 任务 ， 打 开 项 目 配置 文件 ， 设 置 计划 任务 执行 时 间 ， 如 以 
下 代码 所 示 。 
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'CRON MAN TIME'=>360, 

CRON MAX TIME 配置 项 表示 计划 任务 的 执行 时 间 ， 以 秒 为 单位 《〈 即 服务 砸 时间 加 上 
CRON MAX TIME 配置 时 间 等 于 计划 任务 执行 时 间 )。 配 置 完 成 后 在 tags.php 文件 中 定位 
CronRun 行为 ， 如 以 下 代码 所 示 。 

Pa 

通过 前 和 面 的 配置 ， 计 划 任 务 成 功 开 局 。 接 下 来 需要 配置 计划 任务 列表 ，CronRun 使 用 
~crons.php 存放 任务 列表 ， 该 文件 需要 存放 于 项 目 Runtime 目录 中 。 接 下 来 配置 一 条 计划 任 
务 ， 如 以 下 代码 所 示 。 

<?php 

return array ( 

"TESTE rOn! =>array (1 cest ; 3071353052449); 


Jg 
P 


如 上 述 代 码 所 示 ，testeron 表示 任务 名 称 。 对 应 的 值 为 一 维 数组 ， 其 中 testphp 表示 执行 
的 PHP 文件 名 称 ， 该 文件 必须 保存 于 项 目 Lib/Cron 目录 ; 参数 30 表示 执行 间隔 时 间 ; 
1353052449 ZR PRATHER, INE. 

最 后 只 需要 在 Lib/Cron 目录 中 创建 test.php 文件 即 可 。 为 了 方便 演示 ， 这 里 只 需要 在 该 
文件 中 输出 字符 串 即 可 ， 如 以 下 代码 所 示 。 


<?php 
echo "你 好 ， 欢 迎 光 临 本 站 ， 现 在 时 间 是 : " .date("Y-m-d H:i:s",time()); 
2 


在 实际 应 用 开发 中 ， 可 以 在 任务 执行 脚本 中 调用 远程 数据 、 执 行 PHP 支持 的 功能 操作 。 

5. RobotCheck 防 止 数据 把 虫 访问 》 

RobotCheck 行为 用 于 检测 访问 来 源 是 否 是 数据 采集 之 类 的 上 自动 化 软件 ， 从 而 减少 系统 
资源 占用 。RobotCheck 行为 共 文 持 一 项 配置 项 ， 如 以 下 代码 所 示 。 

CT 

开启 RobotCheck 行为 后 ， 只 需要 在 tags.php 文件 中 指定 运行 阶段 中 可 ， 如 以 下 代码 所 示 。 


"app init'=>array ("RAOoCnece"), 


11.4.3 BEXITÄH R 

WAWATA, AR RA OCA Ree ISS UU e rr Du REREH. ThinkPHP II Le 
束 是 扩展 灵活 ， 同 样 行为 也 是 可 以 上 日 定义 扩展 的 。 利 用 目 定 义 行为 扩展 ， 能 够 使 编程 更 加 灵 
活 ， 因 为 扩展 是 一 种 拦截 机 制 ， 一 旦 开局 整个 项 目 都 受 其 约束 ， 这 个 过 程 开 发 人 员 不 需要 在 
项 目 控制 器 或 类 库 中 编写 任何 代码 ， 只 需要 在 配置 文件 中 简单 地 配置 即 可 。 接 下 来 将 深入 介 
绍 日 定义 行为 扩展 的 实现 。 

1. 创建 自 定 义 行为 

日 定义 行为 类 的 定义 与 普通 类 库 及 控制 占有 较 大 不 同 ， 目 定义 行为 类 库 虽 然 也 是 PHP 
功能 类 ,但 目 定 义 行为 只 有 一 个 入 口 函 数 run, EH options 成 员 属 性 映射 项 目 配置 项 ， 如 以 
下 代码 所 示 。 


<?php 
class AutoCheckUserLoginBehavior extends Behavior { 
// 行为 参数 定义 


protected! $options = array l 
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'AutoCheckUserLogin' => false, 
); 
/大 大 
17 
=x Haee Behaviors run (|) 
Ww 
public function run(gSparams ) { 
if(C('AutoCheckUserLogin')) { 
Şthis=>isLogin () s 
}else{ 
FEEUEN True} 
} 
} 
/** 
* 目 动 检测 方法 


protected function isLogin() 1 


if (!session("username")) { 
exit (" 请 登录 后 再 操作 . . .") ; 
} 
} 


} 
2> 


上 述 代 码 是 一 个 名 为 AutoCheckUserLogin 的 目 定 义 行为 类 。 其 中 options 成 员 属 性 是 可 
选项 ， 但 却 是 重要 的 配置 项 ，options 成 员 属 性 值 将 被 系 统 映 射 为 项 目 配置 文件 中 的 配置 项 。 
例如 上 面 的 AutoCheckUserLogin 数组 项 ， 在 运行 时 系统 会 将 该 元 素 映 射 为 配置 文件 中 的 
AutoCheckUserLogin 配置 项 ， 在 项 目 中 或 者 行为 方法 里 直接 使 用 C 函数 即 可 获得 配置 项 值 。 

run 方法 是 行为 类 中 最 关键 的 方法 ， 也 是 普通 类 与 行为 类 最 明显 的 区 别 ，run 方法 是 行为 
关 的 入 口 ， 相 当 于 普通 美的 构造 函 数 ， 所 以 在 创建 行为 关 时 只 需要 该 方 法 设 为 Public 即 可 。 
最 后 还 需要 硼 你 上 日 定义 行为 类 继承 于 Behavior XK. 

和 目 定义 行为 类 创建 完成 后 ， 只 需要 将 该 文件 保存 到 项 目 Lib/Behavior 目录 中 ， 并 以 “ 行 
为 名 + Behavior+.class.php” 的 方式 命名 文件 即 可 。 这 样 一 个 名 为 AutoCheckUserLogin 的 目 
定义 行为 就 可 以 直接 使 用 了 。 

2. 使 用 目 定 义 行 为 

创建 完 日 定义 行为 后 怎么 使 用 ， 这 是 初学 者 容易 困惑 的 问题 。 事 实 上 日 定义 行为 不 需要 
写 任 何 调用 代码 ， 只 需要 配置 即 可 。 继 续 以 AutoCheckUserLogin 行为 为 例 ， 要 使 用 该 行为 只 
需要 两 步 。 首 先 在 tags.php 文件 中 指定 行为 的 运行 阶段 (参照 图 11-8)， 如 以 下 代码 所 示 。 


'app init'=>array ('AutoCheckUserLogin'), 

因为 在 设计 行为 时 指定 了 只 有 配置 参数 AutoCheckUserLogin 为 true 时 才 执 行 用 户 检 
查 ， 所 以 还 需要 在 配置 文件 中 开局 AutoCheckUserLogin， 如 以 下 代码 所 示 。 

"AutoCheckUserLogin"=>true, 

事实 上 ，AutoCheckUserLogin 配置 项 并 不 是 必需 的 ， 也 束 是 说 在 配置 时 ， 目 定义 行为 束 
已 经 能 够 正常 运行 了 ， 配 置 项 只 不 过 用 于 实现 灵活 的 配置 。 通 过 前 面 的 步 坚 ， 现 在 访问 项 目 
任意 一 个 页 面 ， 都 会 执行 用 户 登 录 检测 。 

行为 的 调用 《〈 即 触发 ) 并 非 只 有 配置 tags.php 文件 一 种 方式 ， 但 无 论 从 灵活 性 还 是 方便 
后 期 维护 考虑 ， 使 用 tags.php 文件 配置 都 具备 优势 ， 所 以 这 里 不 再 对 其 他 调用 方式 进行 介 
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绍 ， 感 兴趣 的 读者 可 以 参阅 官方 提供 的 开发 手册 。 

通过 上 述 操 作 ， 可 以 看 到 目 定 义 行为 的 使 用 与 内 置 行为 的 使 用 并 无 区 别 。 同 时 行为 机 制 
的 引入 ， 对 项 目 整 体 控制 提供 了 闹 效 、 灵 活 的 处 理 方 式 。 

3. 配置 自 定义 行为 标签 

可 向 介绍 过 ， 标 签 开 是 行为 的 别名 ， 一 个 标签 可 以 对 应 多 个 行为 。 但 这 些 标签 都 是 系统 
内 置 的 ， 用 于 表示 程序 〈 所 有 基于 ThinkPHP 构造 的 应 用 ) 不 同 的 运行 阶段 ， 事 实 上 开发 人 
员 也 可 以 目 定 义 标 签 ， 用 于 表示 当前 项 目 或 控制 器 的 不 同 运行 阶段 。 假 设 需要 在 当前 控制 大 
载 入 时 执行 前 面 创建 的 AutoCheckUserLogin 行为 ， 代 人 码 如 下 所 示 。 


<?php 
class IndexAction extends Action { 


Publie function initialize (){ 
tag ("incdex init™) p 
} 
} 
>> 
上 述 代 人 码 表示 在 Index 控制 套 初 始 化 时 添加 一 个 “index_init” 行 为 标签 。 讨 加 标签 后 ， 
AutoCheckUserLogin 行为 触发 标签 不 再 是 “app init” 而 是 “index init”， 如 以 下 代码 所 示 。 
El 1nit'=>array('"AutoCneckUseruoogln" ), 
此 时 ， 再 次 访问 项 目 ， 上 只 有 Index 控制 器 下 的 页 耐 受 AutoCheckUserLogin 目 定 义 行为 影 
啊 ， 其 他 页 面 则 不 受 影响 。 


O ”提示 : 如 果 需 要 让 该 标签 约束 当前 项 目 ， 可 以 将 标签 定义 在 公共 控制 器 中 ， 项 目下 的 控制 器 继续 用 该 公 
共 控 制 器 即 可 。 同 样 的 原理 ， 自 定义 标签 也 可 以 定义 在 控制 器 前 操作 或 后 操作 方法 中 ， 实 现 单个 动作 页 
面 的 行为 控制 。 


11.5 ”小结 


KEEN S ThinkPHP 内 置 的 扩展 功能 ， 包 括 模式 扩展 、 模 板 扩展 、 类 库 扩 展 、 行 
为 扩展 、 标 签 扩展 。 其 中 行为 扩展 是 一 种 全 新 的 概念 ， 在 前 面 的 章节 内 容 中 均 没 有 涉及 相关 
介绍 ， 在 本 章 里 通过 直观 的 图 形 ， 简 单 的 示例 ， 帮 助 读者 快速 掌握 行为 扩展 的 实际 应 用 。 

ThinkPHP 的 灵活 之 处 在 于 扩展 丰 主 ， 开 发 人 员 不 仅 可 以 使 用 系统 内 置 的 高 效 扩展 ， 还 
可 以 利用 现 有 的 PH 类 ， 直 接 以 第 三 方 扩展 形式 整合 到 系统 中 。 

本 草 重 点 选取 了 类 库 扩展 进行 深入 介绍 。 使 用 第 三 方 类 库 ， 使 得 ThinkPHP 功能 得 到 无 
限 的 扩展 。 本 革 所 介绍 的 扩展 类 库 读者 均 可 以 在 http://www.beauty-soft.net/book/php_mvc/ 网 
站 下 载 。 

PIE, AXR ThinkPHP 功能 就 全 部 介绍 完了 。 后 和 面 的 内 容 将 继续 深入 探讨 PHP, Zem Ob 
述 当 前 网 站 开发 中 最 主流 的 技术 ， 通 过 这 些 谍 程 的 学 习 ， 读 者 完全 可 以 使 用 PHP MVC 开发 
各 类 大 中 型 网 站 。 
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内 容 提要 


分 布 式 开发 是 一 项 用 于 实现 各 种 软件 硬 平台 交互 的 技术 。 它 不 同 于 Json 或 者 
XML， 分 布 式 开发 更 注重 服务 调用 。 在 分 布 式 开发 中 ， 最 第 用 的 技术 有 .NET Remoting、 
WCF, Web Server, # + Web Server 使 用 的 是 W3C Se WSDL 数据 格式 。 

本 章 首 先 介绍 SOAP 与 SOA 的 概念 ， 然 后 对 SOAP 的 消息 体 进 行 重点 介绍 ， 接 着 还 会 
对 Zend Studio 可 视 化 创建 SOAP 进行 讲解 ， 最 后 介绍 Web Service 的 性 能 测试 。 阅 读本 章 需 
要 读者 具备 XML 基础 知识 。 


学 习 目 标 


@ 了 了 解 扩 展 概念 。 

@ 了 解 PHP 对 SOAP 的 支持 特性 。 

@ 理解 SOAP 与 SOA- 

@ HUR WSDL 组 成 格式 。 

@ 掌握 WSDL 消 县 体 季 点 的 创建 。 

@ 掌握 可 视 化 创建 WSDL 的 步骤 。 

@ 掌握 nusoap 套件 的 使 用 。 

© 学 握 在 ThinkPHP 中 开发 Web Service 的 方法 。 
@ 钛 悉 soapUI 测 试 工具 的 使 用 。 
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12.1 分 布 式 开发 概念 


Web Service 是 分 布 式 开发 中 一 项 非常 重要 的 技术 ， 分 布 式 开发 一 个 重要 的 概念 束 是 将 
服务 放置 在 Web 上 ， 而 客户 端 只 需要 调用 服务 ， 其 他 的 数据 验证 、 协 议 、 数 据 驱 动 等 都 由 
服务 中 心 来 完成 ， 客 户 端 接收 处 理 后 的 结果 即 可 。Web Service 束 是 通过 标准 的 XML 数据 格 
式 和 通用 的 互联 网 协议 为 其 他 应 用 程序 提供 服务 的 。 

在 为 其 他 应 用 程序 提供 服务 时 ，Web Service 可 以 以 接口 的 方式 接收 合法 的 请 求 ， 并 返 
回 相 应 的 服务 和 功能 。 应 用 程序 在 获取 这 些 特定 的 服务 和 功能 时 ， 只 二 实现 这 些 接 口 即 可 完 
成 数据 则 的 交互 。 分 布 式 开发 没有 统一 的 服务 主体 ， 只 要 服务 方 这 循 SOA (Service-Oriented 
Architecture) 服务 契约 ， 束 可 以 癌 客 户 疹 提供 服务 。 

SOA 是 一 种 摘 述 分 布 式 开发 的 架构 ， 能 够 将 不 同 的 服务 通过 定义 好 的 接口 和 殷 约 相互 
关联 起 来 。 下 面 首先 介绍 SOA 与 SOAP 之 间 的 关系 。 


12.1.1 SOA 5 SOAP 


SOA ZS ERAAN, E ERR S AFR m e AE Ae e 
21. SOA 不 属于 任何 厂商 ， 它 是 独立 于 第 三 方 的 。 虽 然 实 现 分 布 式 开发 的 不 仅 只 有 SOA, 
但 SOA 是 一 种 国际 化 标准 ， 常 见 的 ASMX, WSDL, CORBA (Common Object Request 
Broker Architecture) 等 主流 分 布 式 开发 技术 都 是 基于 SOA 规范 的 。 

SOA 组 件 之 间 进 行 通信 时 ， 使 用 SOAP 协议 进行 传递 。 它 最 先 由 微软 公司 于 1999 年 发 
布 并 整合 到 .NET Framework 1.0 中 ，2001 年 被 W3C 正式 定义 为 一 种 轻 量 的 、 人 简单 的 、 基 于 
XML 的 通信 协议 。 经 过 十 多 年 的 发 展 ， 微 软 公司 基于 SOAP 1.2 通信 协议 发 展 和 完善 了 分 布 
式 开 发 技术 ， 最 新 的 演进 技术 为 WCF (Windows Communication Foundation )。 

SOAP 类 似 于 HITP， 在 通信 时 使 用 HTTP 发 送 XML 格式 数据 ， 使 用 RPC 调用 远程 结 
果 ， 所 以 傈 单 理解 束 是 HTTP 和 XML 两 种 通信 协议 的 结合 。 此 外 ，SOAP 不 仅 文 持 HTTP 
还 文 持 SMTP, MIME 等 多 种 成 熟 的 网 络 传输 协议 ， 也 束 是 说 使 用 SOAP 协议 可 以 代替 
HTTP, SMTP 等 受 文 持 的 协议 。 

网 络 上 各 种 Web 服务 器 都 可 以 作为 SOAP 的 平台 ， 各 服务 器 平台 只 需要 生成 或 创建 符 
F SOAP 协议 的 服务 ， 即 可 在 各 种 客户 问 间 相互 调用 。 比 如 客户 端 调 用 服务 方 的 服务 ， 只 般 
要 在 这 些 Web 服务 右上 创建 服务 ， 然 后 公开 SOAP 规范 接口 ， 残 能 够 进行 调用 。 

Web Service 平台 需要 一 僚 协 议 来 实现 分 布 式 应 用 程序 的 创建 。 任 何平 台 都 有 它 的 数据 
表示 方法 和 类 型 系统 。 要 实现 互 操作 性 ，Web Service 平台 必须 提供 一 套 标准 的 类 型 系统 ， 
用 于 沟通 不 同 平台 、 编 程 语言 和 组 件 模 型 中 的 不 同类 型 系统 。 

Web Service 束 是 使 用 SOAP 协议 ， 实 现 各 服务 间 互 相 调 用 的 平台 。 客 户 并 调用 Web 
Service 只 需要 生成 一 个 代理 ， 然 后 声明 需要 调用 的 服务 公开 方法 ， 最 后 使 用 寞 步 通 信 返 回 请 
求 的 数据 ( 即 服务 结 末 )， 如 图 12-1 所 示 。 

PHP 5.X 内 置 了 SOAP 扩展 ， 能 够 同时 支持 客户 端 访 问 模式 及 远程 服务 模式 。 其 中 在 客 
Pin PRIN F, PHP 使 用 SoapClient 初始 化 连接 ， 在 服务 模式 时 ，PHP 使 用 SoapServer 初 
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始 化 SOAP。 无 论处 于 哪 种 模式 ，PHP 都 是 使 用 WSDL 数据 格式 进行 通信 的 。 


据 
Pä 
' 


图 12-1 SOAP 通信 过 程 


如 图 12-1 所 示 ， 读 者 也 许 会 感觉 到 整个 流程 与 普通 的 XML 通信 过 程 并 无 区 别 ， 事 实 
上 Web Service 整个 开发 框 保 核心 在 SOAP 协议 。 假 设 以 传统 的 HTTP 访问 WSDL, e 
WSDL 束 是 一 个 XML 文件 ， 那 么 浏览 器 也 不 会 得 到 运算 结果 以 源 代 人 码 显示 )。 但 要 是 在 
文 持 SOAP 访问 协议 的 容 吉 中 访问 WSDL 文件 时 ， 将 会 得 到 SOAP 服务 ， 并 能 够 发 现 被 注 
册 的 公开 服务 。WSDL 中 定义 的 XML 区 点， 并 不 是 访问 就 会 被 触发 的 ， 需 要 在 应 用 程序 中 
使 用 UDDI (Universal Description, Discovery and Integration， 提 供 基 于 Web 服务 的 注册 和 发 
现 机 制 ) 登记 捷 需 要 请 求 的 服务 节点 。 

XEMEN A miee A Web Service 分 布 式 开发 ， 关 键 是 看 客户 疹 文 不 文 持 
SOAP 通信 协议 。 对 于 服务 病 而 言 ， 文 不 文 持 分 布 式 开发 服务 ， 关 键 是 看 解释 引擎 有 没有 所 
供 SOA 生成 模块 。 SOA 虽然 是 一 套 W3C 规范 ， 但 是 并 不 意味 着 各 种 解释 引擎 都 能 够 完美 
文 持 ， 这 也 就 造成 各 种 开发 技术 的 功能 实现 不 尽 相 同 。 

例如 在 CH Java 中 ， 对 SOA 的 文 持 是 最 强大 和 最 完整 的 ， 不 仅 文 持 动 态 生 成 ， 而 且 文 
持 数据 流通 信 、 长 文本 等 。 而 在 PHP 中 只 文 持 表态 定义 而 且 不 文 持 数据 流 ， 在 PHP 5.4 以 上 
版 本 文 持 得 比较 好 。 

对 于 开发 者 而 言 ，XML 需要 开发 人 员 上 自己 手动 编写 处 理 过 程 ， 根 据 XML 解释 器 的 差 
卉 ， 处 理 的 手段 也 不 一 样 ， 过 程 也 相对 比较 复杂 。 而 使 用 SOAP 时 ， 开 友人 员 不 需要 手动 处 
理解 释 过 程 ， 因 为 SOAP 解释 器 已 经 日 动 完成 所 有 的 数据 解释 工作 ， 需 要 做 的 就 是 调用 
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WSDL 文档 中 的 服务 ( 即 节 点 ， 通 常 一 个 节点 对 应 一 个 类 成 员 方 法 )， 而 且 无 论 使 用 什么 技 
本， 过 程 都 是 统一 的 ， 不 会 造成 差异 。 在 开发 形式 上 与 本 地 调用 功能 次 并 无 区 别 ， 所 以 开发 
效率 上 无 疑 比 XML 更 高 。 

与 HTTP 一 样 ， 无 论 在 哪 种 语言 技术 中 ，SOAP 都 是 异步 通信 的 ， 所 以 能 够 很 好 地 解决 
性 能 问题 ， 尤 其 像 手机 一 类 的 内 存 受 限 的 终端 应 用 开发 ， 使 用 SOAP JEP áo Web Service 
开发 虽然 有 很 多 优点 ， 但 缺点 也 非常 明显 ， 尤 其 在 PHP 中 。 下 面 将 结合 PHP AR, EMN 
绍 在 PHP 中 的 文 持 情况 ， 然 后 再 详细 介绍 WSDL 文档 。 


12.1.2 PHP 5 分 布 式 开发 


PHP 5 以 插件 的 形式 提供 SOAP 服务 ， 所 以 在 编 详 时 ， 和 需要 加 入 --enable-soap 选项 。 使 
用 phpinfo 输出 信息 时 ， 如 果 存 在 SOAP 选项 ， 表 示 服 务 已 经 生效 ， 如 图 12-2 rz, 


soap.wsdl_cache_enabled 
EE e 


图 12-2 SOAP 服务 


如 末 服 务 没有 开局 ， 需 要 在 PHP 配置 文件 中 开局 ， 如 以 下 代码 所 示 。 


;extension=php_snmp.dll 
extension=php_soap.dll 


在 Linux 操作 平台 开启 SOAP 服务 的 代码 如 下 所 示 。 


;extension=php_snmp.dl11 


extension=php_soap.so 
SOAP 有 目 己 的 配置 选项 ， 如 以 下 代码 所 示 。 


[soap] 


; Enables or disables WSDL caching feature. 
; http://php.net/soap.wsdl-cache-enabled 
soap.wsdl_cache_enabled=0 


; Sets the directory name where SOAP extension will put cache files. 

; http://php.net/soap.wsdl-cache-dir 

soap.wsdl_cache_dir="/tmp" 

; (time to live) Sets the number of second while cached file will be used 
; instead of original one. 

; http://php.net/soap.wsdl-cache-tt1 

soap.wsdl_cache_tt1=86400 


; Sets the size of the cache limit. (Max. number of WSDL files to cache) 


soap.wsdl_cache_limit = 5 
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其 中 选项 soap.wsdl cache enabled 表示 是 否 关 闭 WSDL 数据 缓存 ， 值 为 1 时 禁用 绥 存 ， 
为 了 方便 程序 调试 ， 在 开发 阶段 建议 将 该 项 设置 为 1; soap.wsdl cache dir 表示 绥 存 存放 有 目 
录 ，Windows 系统 可 以 设置 为 c:\tmp; soap.wsdl cache ttl 表示 绥 存 生命 周期 ， 以 秒 为 单位 ， 
默认 86400; soap.wsdl cache limit 表示 绥 存 数据 大 小 ， 以 MB 为 单位 。 

配置 文件 修改 后 ， 重 启 PHP 服务 或 者 php-fpm， 即 可 生效 。 如 果 PHP 运行 于 非 安全 模 
式 下 ， 上 述 配 置 项 可 以 直接 在 PHP 代码 中 使 用 ini set 设置 ， 例 如 关闭 WSDL 缓存 功 能 ， 如 
以 下 代码 所 示 。 

ini_set ("soap.wsdl_ cache enabled", "1"); 

剖面 提 人 到 过 PHP 既 可 以 作为 SOAP 的 客户 蹦 ， 也 可 以 作为 服务 疹 。 作 为 服务 靖 时 ， 使 
用 SoapServer 实例 类 初始 化 SOAP 服务 ， 如 以 下 代码 所 示 。 


Şserver =new SoapServer ('./wsdl/UserDataSoap.wsdl',array ('soap_version' => 
SOAP_1_2)); 


其 中 参数 1 表示 SOA 服务 描述 文件 ， 即 WSDL 文件 ， 该 文件 是 SOAP 通信 的 核心 ， 参 
数 2 表示 SOAP 协议 版 本 ， 常 用 的 版 本 有 1.0 及 1.2， 本 章 内 容 及 示例 全 部 基于 SOAP 1.2 版 
本 实现 。 

得 到 SoapServer 实例 对 象 ， 就 可 以 调用 对 和 象 中 的 setClass 方法 设置 WSDL 所 描述 的 消息 
服务 。 通 单 情况 下 ， 需 要 将 setClass 指定 实例 类 摘 述 为 可 公开 调用 的 服务 接口 。 假 设 需要 将 
User KHR WSDL 服务 接口 ， 代 码 如 下 所 示 。 


public function SoapService(){ 


ini_set ("soap.wsdl_cache_enabled", "1"); // disabling WSDL cache 
$server =new SoapServer('./wsdl/User.wsdl',array ('soap_version' => SOAP_1_2)); 
Şserver—->set 
SEET Ee 
$server->setClass ("User"); 
if (isset ($SHTTP RAW POST DATA)) { 
srequest = S$HTTP_ RAW_ POST_ DATA: 
} else { 
Srequest = file_get_contents ('php://input'); 
} 


Şserver—->handle ($request); 


i 
user.php 美文 件 如 以 下 代码 所 示 。 
<?php 
elass User 
public function getUser ($username) { 
return $username; 
| 
} 


?> 


所 有 作为 公开 调用 的 服务 接口 ， 成 员 类 方法 必须 修饰 为 public 权限 。 
当 PHP 作为 客户 问 模 式 时 ， 使 用 SoapClient 实例 类 。 假 设 需要 调用 前 面 创建 的 user DR 
务 ， 代 码 如 下 所 示 。 


<?php 


$soap = new SoapClient ('./wsdl/user.wsdl');}; 
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$rows=$soap-> getUser (" 李 开 涌 ") ; 


var_dump ($rows); 


在 实际 应 用 开发 时 ，WSDL 文件 可 以 是 远程 的 文件 ， 并 且 支 持 跨 域 访问 。 可 以 看 到 ， 
PHP 在 调用 SOAP 服务 时 ， 开 友人 员 不 需要 编写 任何 处 理 WSDL 或 者 XML 文档 的 过 程 ， 只 
需要 像 本 地 开发 一 样 调用 对 象 中 的 方法 即 可 。 
看 到 这 相信 读者 已 经 感觉 到 分 布 式 开发 无 论 是 客户 端 还 是 服务 端 ，PHP 所 需要 做 的 工作 
非常 少 。 关 键 是 WSDL 文档 模型 ， 这 里 并 没有 涉及 相关 内 容 ， 是 因为 WSDL 确实 是 分 布 式 
开发 中 最 重要 的 文档 模型 ， 读 者 只 有 彻底 理解 该 文档 模型 ， 才 能 正确 开发 分 布 式 应 用 ， 所 以 
接 下 来 的 内 容 将 重点 围绕 该 文档 展开 。 
虽然 SOAP 开发 具有 很 多 优点 ， 但 并 非 没 有 缺点 ， 在 实际 开发 中 应 该 根据 需要 考虑 是 否 
该 使 用 SOAP RE XML 或 者 Json， 存 在 的 缺点 主要 如 下 。 
> 在 PHP 中 内 置 的 SOAP 扩展 并 不 支持 生成 生成 SOA 服务 ， 开 发 人 员 需 要 手动 创建 
WSDL 文件 ， 而 Json 或 XML 等 ，PHP 直接 提供 生成 及 解释 功能 。 

> WSDL 文件 虽然 是 SGML 文件 ， 但 需要 使 用 SOAP 协议 访问 ， 如 果 使 用 普通 的 浏览 
器 访问 WSDL 文件 ， 将 不 能 触发 相应 服务 。 这 就 意味 着 在 开发 阶段 要 进行 调试 ， 过 
程 将 变 得 复 淋 。 使 用 普通 的 Firebug 等 插件 不 能 获取 到 寞 音信 息 ， 而 断 点 调试 是 PHP 
的 弱项 ， 所 以 调试 问题 将 会 是 SOAP 开发 首要 面 对 的 问题 。 

> 能 不 能 使 用 SOAP， 关 键 是 看 解释 引擎 文 不 文 持 SOAP 协议 及 SOA 服务 。 虽 然 主 沉 
的 技术 ， 例 如 Object-c7 CH, C++, Java 等 都 提供 了 良好 的 支持 ， 但 常见 的 
JavaScript 并 不 支持 SOAP. 


12.2 SOAP 消息 体 


SOAP 服务 需要 由 WSDL 描述 及 注册 ，WSDL 的 全 称 是 Web Services Description 
Language， 是 用 于 描述 Web 服务 通过 XML 与 客户 端 通信 的 一 套 文 档 模型 。 这 里 的 Web DR 
务 通 第 就 是 指 网 站 程序 中 的 功能 模块 ， 例 如 功能 类 ， 函 数 等 都 可 以 作为 接口 。 所 以 WSDL 
也 称 SOAP 消息 体 ， 对 于 开发 者 而 言 ， 该 文档 并 不 阳 生 ， 因 为 与 名 匈 的 XML 文档 类 似 。 接 
下 来 将 对 WSDL 文档 模型 进行 详细 介绍 。 


12.2.1 WSDL 文件 


在 PHP F, SOAP 扩展 模块 只 能 解释 WSDL 文件 ， 这 也 是 W3C 所 规范 的 文件 类 型 。 事 
SKE WSDL 并 非 唯一 的 SOAP 文档 模型 ， 在 其 他 平台 中 出 于 商业 需要 ， 各 三 商会 进行 相应 
的 功能 定制 ， 文 件 名 称 也 不 尽 相 同 ， 例 如 ASP.NET FRW ASMX. 

但 不 管 怎样 叫 法 ， 由 于 WSDL 遵循 国际 化 规范 ， 所 以 在 这 些 平台 上 使 用 WSDL 是 完全 
通用 的 。WSDL 由 definitions, types, message, binding, service 等 节点 组 成 ， 分 别 代 表 不 同 
的 描述 信息 ， 如 下 所 示 。 

> types QÏ AKW): 可 选 ， 数 据 类 型 定义 的 容器 ， 它 使 用 菏 种 闫 型 系统 《如 XSD). 
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> message ĦA): 必 选 ， 通 信和 数据 的 抽象 类 型 化 定义 ， 它 由 一 个 或 者 多 个 part 
组 成 。 
part: 必 选 ， 消 县 参数 。 
operation 〈 操 作 ): 必 选 ， 对 服务 所 文 持 的 操作 进行 抽象 摘 述 。 
port Type mO): 必 选 ， 特 定 端口 类 型 的 具体 协议 和 数据 格式 规范 。 
binding (通信 类 型 绑 定 ):， 必 选 ， 描 述 通 信 的 类 型 ， 例 如 SOAP. 
service〈 服 务 描述 ): 必 选 ， 相 关 奖 口 的 集合 ， 包 括 其 关联 的 接口 、 操 作 、 消 息 等 。 
port《〈 服 务 列表 ): 必 选 ， 定 义 为 绑 定 和 网 络 地 址 组 合 的 单个 山 点 。 

其 中 operation 操作 分 为 4 种 类 型 ， 分 别 为 one-way 〈 单 癌 )、request-response 〈 异 步 请 求 
异步 返回 ) solicit-response (要 求 应 答 )、notification (通知 )。WSDL 最 外 层 元 素 是 
definitions， 在 该 元 素 下 定义 其 他 市 点 插 述 信息 。 如 图 12-3 fro 


VW WW WM Y 


Note: AN clements excepi <import>, <indude*, 
and <iypes> can have choaturez andlor 
<property> as children 


图 12-3 WSDL 文档 模型 


如 图 12-3 所 示 ， 这 些 节 点 描述 信息 并 非 全 是 必 选 的 ， 下 面 将 通过 一 个 示例 代码 ， 人 简单 
演示 SOAP 的 应 用 。 假 设 使 用 http://cl.localhost 作为 Web Service HRS mi, MEH 
http://c2.localhost 作为 客户 关 ， 那 么 在 服务 关中 需要 创建 消息 实体 类 。 首 先 创建 绑 定 接口 文 
件 ， 并 命名 为 apiphp， 代 码 如 下 。 


<?php 
include ("Data.class.php"); 


Imi ser (VeoaD. wee eacha enable, WLW) g 
$server =new SoapServer ('./api.wsdl',array ('soap_version'! => SOAP_1_2)); 
$server->setClass ("Data"); 
if (isset ($HTTP_ RAW POST DATA)) { 
srequest = S$HTTP_ RAW_ POST_ DATA: 


LI else? 
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$request = file_get_contents ('php://input'); 
} 


sserver->handle ($request); 


对 应 的 Data.class.php 文件 为 消 居 类 体 文件 ， 代 人 码 如 下 所 示 。 
<?php 


class Data{ 
public function getData (Scity)t 

Şarray=array ( 
m100000"™=>" 北 京 "， 
no00000n mE 
REENEN 
ëoünO-s Wält, 
"510000"=>" Me, 

); 


ecuen Sarray l Seicy] y 


} 


上 述 代码 功能 很 简单 ， 只 需要 根据 客户 端 提交 的 邮政 编码 ， 返 回 相应 的 城市 。 在 实际 应 
用 开发 中 ， 这 些 数据 可 以 直接 在 数据 库 中 获取 。 前 面 在 apiphp 文件 中 已 经 将 api.wsdl 绑 定 
为 消息 类 体 描述 文件 ， 所 以 需要 在 同 级 目录 下 创建 apiwsdl 文件 ， 代 码 如 下 。 


<?xml version="1.0" encoding="UTF-8" standalone="no"?> 


<wsdl:definitions xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" 
xmlns:tns="http://www.example.org/api/" xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" 
xmlns:xsd="http://www.w3.o0org/2001/XMLSchema" name="api" 


targetNamespace="http://www.example.org/api/"> 
<! 一 消息 类 型 列表 --> 
<wsdl:types> 
<xsd:schema targetNamespace="http://www.example.org/api/"> 
<! 一 上 传 消 轧 类 型 --> 
<xsd:element name="getData"> 
<xsd:complexType> 
<xsd:sequence> 
<xsd:element name="in" type="xsd:string"/> 
</xsd: sequence> 
</xsd:complexType> 
</xsd:element> 
<! 一 返回 消 轧 类 型 --> 
<xsd:element name="ReposDatagetDataResponse"> 
<xsd:complexType> 
<xsd:sequence> 
<xsd:element name="out" type="xsd:string"/> 
</xsd: sequence> 
</xsd:complexType> 
</xsd:element> 
</xsd:schema> 
</wsdl:types> 


<! 一 公开 的 消息 列表 ， 一 组 消息 类 型 由 上 传 及 获取 两 个 Part 构造 --> 
<wsdl:message name="getDataRequest"> 
<wsdl:part name="city" type="xsd:string"/> 


</wsdl:message> 
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<wsdl:message name="getDataResponse"> 
<wsdl:part name="ResponseData" type="xsd:string"/> 
</wsdl:message> 


<! 一 端口 类 型 ， 只 有 加 入 portType 列表 的 方法 才 可 以 对 外 公开 调用 --> 
<wsdl:portType name="api"> 
<wsdl:operation name="getData"> 
<wsdl:input message="tns:getDataRequest"/> 
<wsdl:output message="tns:getDataResponse"/> 
</wsdl:operation> 
</wsdl:portType> 
<! EIR --> 
<wsdl:binding name="apiSOAP" type="tns:api"> 
<soap:binding style="document" transport="http://schemas .xmlsoap.org/soap/http"/> 
<wsdl:operation name="getData"> 
<soap:operation soapAction="http://www.example.org/api/getData"/> 
<wsdl:input> 
<soap:body use="literal"/> 
</wsdl:input> 
<wsdl:output> 
<soap:body use="literal"/> 
</wsdl:output> 
</wsdl:operation> 
</wsdl:binding> 
<! 一 服务 描述 ， 即 后 合 PHP--> 
<wsdl:service name="api"> 
<wsdl:port binding="tns:apiSOAP" name="apiSOAP"> 
<soap:address location="http://c1.localhost/api.php"/> 
</wsdl:port> 
</wsdl:service> 
</wsdl:definitions> 


如 上 述 代 码 所 示 ，types TREX T WAKAR (Ek KiB). PH 目前 只 文 持 
string, int, float 等 常见 数据 类 型 。 这 里 的 数据 类 型 与 PHP 中 的 数据 类 型 不 是 一 个 概念， 
是 W3C 定义 的 数据 类 型 。 也 束 是 说 在 此 定义 的 数据 类 型 无 论 在 哪 种 语言 或 技术 中 ， 都 能 够 
转换 为 该 语言 对 应 的 数据 类 型 。 对 于 PHP 而 言 ， 由 于 PHP 对 数据 类 型 不 敏感 ， 所 以 意义 不 
大 《例如 可 以 使 用 string RE int). 

至 此 ， 一 个 人 简单 的 SOAP 服务 端 驶 创建 完成 了 。 接 下 来 在 http:/c2.localhost 网 站 中 创建 
客户 器 ， 代 但 如 下 上 所 示 。 

<?php 

/ 1 im 

$soap = new SoapClient ('http://cl.localhost/api.wsdl'); 


Şrows=$soap->getData ("510000"); 


var_dump ($rows); 


运行 结果 为 “广州 市 ”可 以 看 到 ， 在 客户 端 中 并 不 是 调用 apiphp 文件 ， 而 是 调用 
api.wsdl 文件 。api.wsdl 文件 是 一 个 XML 格式 的 数据 描述 文件 ， 并 没有 脚本 处 理 能 力 。 但 由 
于 将 其 与 apiphp 文件 绑 定 ， 所 以 api.wsdl SFWA T apiphp 所 拥有 的 功能 了 ， 在 使 用 时 
与 在 本 地 直接 调用 api.php 文件 保持 一 致 ， 这 就 是 分 布 式 开发 的 典型 应 用 。 

需要 注意 的 是 ， 由 于 PHP 对 数据 类 型 不 敏感 ， 所 以 对 调用 的 结果 并 不 会 进行 强 类 型 检 
测 ， 这 往往 会 造成 严重 问题 。 
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例如 在 WSDL 中 定义 了 array 类 型 数据 ， 在 调用 时 PHP 可 以 使 用 array 类 型 传 参 数 给 远 
程 接 口 。 但 使 用 其 他 语言 时 (例如 CA Java) 将 会 造成 严重 的 错误 。 这 是 因为 PHP 对 
WSDL 的 文 持 是 严格 按照 W3C 规范 所 设计 的 ， 而 访 规 范 中 并 未 将 数组 类 型 作为 SOAP 标准 
数据 类 型 〈 需 要 额外 引入 另外 一 组 命名 空间 )。 

所 以 强制 将 节点 中 的 string 改 为 array 时 ，PHP 能 正 第 识别 为 数组 类 型 ， 但 其 他 强 类 型 
语言 则 不 能 识别 。 解 决 办 法 是 尽量 使 用 W3C 规范 的 标准 数据 类 型 ， 例 如 可 以 将 数组 序列 化 
为 JgJon， 然 后 以 字符 串 的 形式 传 参 。 

接 下 来 将 对 WSDL 中 重要 的 节点 元 素 进 行 讲解 。 理 解 这 些 节 点 的 作用 及 意义 ， 是 掌握 
SOAP 应 用 开发 的 前 提 。 首 先 从 definitions 元 素 开 始 介绍 。 


12.2.2 ”定义 根 消 息 体 definitions 


definitions 元 素 是 一 个 标准 的 WSDL 根 元 素 ， 根 元 素 只 能 有 一 个 ， 并 且 成 对 出 现 ， 如 以 
下 代码 所 示 。 


<?xml version="1 .0" encoding="UTF-8" standalone="no"?> 


<wsdl:definitions 
xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" 
xmlns:tns="http://www.example.org/api/" 
xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/" 
xmlns:xsd="http://www.w3.o0org/2001/XMLSchema" 
name="api" 
targetNamespace="http://www.example.org/api/"> 

<! 一 其 他 节点 元 素 一 -> 
</wsdl :definitions> 


如 上 述 代码 所 示 ，definitions 根 元 素 共 定 义 了 6 个 属性 ， 分 别 如 下 : 
xmlns:Soap: 声明 SOAP MAT E. 
xmlns:tns: 使 用 tns 前 级 指 问 日 身 命名 空间 (相当 于 功能 类 中 的 this)。 
xmlns:wsdl: 声明 WSDL 命名 空间 。 
xmlns:xsd: 声明 XML Schema 及 DTD 命名 空间 ， 对 XML 数据 进行 校 验 。 
name: 当前 文档 模型 的 名 称 。 
targetNamespace: 指定 返回 值 的 XML 命名 空间 。 
上 述 文档 定义 ， 除 了 xmlns:wsdl 命名 空间 及 name 属性 是 必 选 项 外 ， 其 他 都 是 可 选 的 。 
但 为 了 更 加 标准 化 ， 建 议 读者 全 部 使 用 。 


VW WW WM Y 


12.2.3 type 类 型 


type 节 扣 用 于 定义 当前 文档 模型 文 持 的 数据 闪 型 。 这 里 的 数据 类型 是 一 种 抽象 的 数据 区 
型 ， 根 据 W3C 的 定义 ， 这 些 数 据 类 型 会 被 客户 问 语 诗 转 换 为 能 够 识别 的 数据 类 型 。 假 设 在 
WSDL 消息 中 定义 一 个 接口 参数 ， 束 必须 要 严格 定义 数据 类 型 。 如 以 下 代码 所 示 。 


<wsdl:types> 


<xsd:schema targetNamespace="http://www.example.org/api/"> 
<xsd:element name="getData_String"> 
<xsd:complexType> 
<xsd:sequence> 
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<xsd:element name="in" type="xsd:string"/> 
</xsd: sequence> 
</xsd:complexType> 
</xsd:element> 
<xsd:element name="Response_int"> 
<xsd:complexType> 
<xsd:sequence> 
<xsd:element name="out" type="xsd:int"/> 
</xsd: sequence> 
</xsd:complexType> 
</xsd:element> 
</xsd: schema> 
</wsdl :types> 


上 述 代 人 码 定义 了 两 个 抽象 的 数据 类 型 ， 即 getData 及 Response string。 所 有 数据 类 型 需 
要 定义 在 xsd:schema 节点 中 ，xsd:schema 是 一 组 类 似 于 DTD 的 XML 文档 标记 验证 模型 ， 
对 于 WSDL 而 言 ， 所 有 数据 次 型 需要 遵 箱 xsd:schema 标准 。 定 义 好 数据 类 型 列表 后 ， 在 使 
用 时 直接 引用 相应 的 市 点 名 称 妈 可。 例如 需要 使 用 string 数据 类 型 代码 如 下 。 


<wsdl:part name="city" type="xsd: getData String "/> 


当然 ， 如 果 不 使 用 节点 作为 数据 类 型 ， 可 以 直接 使 用 xsd 声明 数据 类 型 即 可 ， 如 以 下 代 
ELO 


<wsdl:part name="city" type="xsd:string"/> 


使 用 节点 元 素 作 为 数据 类 型 主要 是 方便 文档 模型 的 管理 ， 以 及 扩展 数据 类 型 (PHP 因为 
不 支持 动态 生成 WSDL， 所 目前 不 支持 数据 类 型 扩展 )， 但 为 了 代码 更 加 简洁， 在 实际 应 用 
开发 中 一 般 直 接 声 明 即 可 。xsd( 妈 xml: schema) 标准 的 数据 类 型 如 表 12-1 所 示 。 


表 12-1 schema 支持 的 标准 数据 类 型 


对 应 Java 数据 类 型 


xsd:boolean 布尔 值 boolean 

xsd:date 日 期 类 型 new Date(); 

xsd:dateTime 可 以 格式 化 的 时 间 日 期 类 型 java.util.Date 

ed: double 双 精 度数 值 double 

xsd:float 浮 点 数值 float 

xsd:hexBinary 二 进 制 数 据 类 型 bytel] 

xsd:int 整 型 数值 int 

xsd:string 字符 串 java.lang.String 

xsd:time UNIX HFJ Javax.xml.datatype.XMLGregorianCalendar 


12.2.4 ”portType 端口 类 型 


portType 六 点 是 WSDL 文档 模型 中 重要 的 节点 元 素 ， 也 是 一 个 必 不 可 少 的 元 素 。 它 完整 
地 摘 述 了 公开 可 调用 的 消 恩 接口 ， 包 括 上 传 消 恩 接口 及 下 行 数据 接口 。 其 中 上 传 消 轧 接口 使 
用 wsdl:input TREX; 下 行 数据 接口 使 用 wsdl:output 元 素 定 义 。 每 个 元 素 定 义 的 值 必 须 是 
当前 文档 模型 中 存在 的 操作 (operation )。 如 以 下 代码 所 示 。 


<wsdl:portType name="api"> 


<wsdl:operation name="getData"> 
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<wsdl:input message="tns:getDataRequest"/> 
<wsdl:output message="tns:getDataResponse"/> 
</wsdl:operation> 
</wsdl:portType> 


如 上 述 代 人 码 所 示 ， 属 性 portType KRAF MOKAER, ee PHP 类 名 ， 后 面 介 
绍 的 binding 节点 类 型 需要 指定 为 该 名 称 。input 的 操作 为 getDataRequest， 事 实 上 当前 文档 
模型 中 并 不 存在 getDataRequest。 真 正 对 应 的 操作 是 getData， 但 portType D ais J H DZ 
型 (由 message 定义 操作 类 型 )。 也 束 是 说 最 终 的 结果 由 message 节点 来 决定 。 如 果 有 多 个 
操作 〈 即 类 成 员 方 法 ) 需要 作为 公开 接口 ， 投 照相 同 的 格式 添加 上 即 可 。 


12.2.5 message 消息 列表 


message 节点 直接 影响 SOAP 交互 的 结果 ， 前 面 的 内 容 已 经 提 到 过 ，portType TAR 
公开 接口 ， 但 接收 数据 上 传 及 数据 返回 ， 将 由 message KEX. message 是 数据 的 抽象 定 
义 ， 它 本 里 不 执行 功能 。 假 设 将 operation 看 作 是 成 员 类 中 的 方法 的 话 ， 那 么 message WN 
法 中 的 参数 ， 如 以 下 代码 所 示 。 


<wsdl:message name="getDataRequest"> 


<wsdl:part name="city" type="xsd:string"/> 
</wsdl :message> 
<wsdl:message name="getDataResponse"> 

<wsdl:part name="ResponseData" type="xsd:string"/> 
</wsdl :message> 


其 中 数据 类 型 可 以 是 type 中 定义 的 节点 元 素 名 称 ， 也 可 以 是 schema 标准 数据 类 型 。 上 
述 代 码 共 定义 了 一 个 参数 ， 即 city 对 应 的 数据 类 型 为 string。 如 果 有 多 个 参数 ， 只 需要 继续 
添加 wsdl:part 节点 元 素 即 可 。message 消息 列 表 将 每 个 操作 (operation) 分 为 接收 模式 
(Request) 及 返回 模式 〈Response)， 以 便 客 户 端 在 异步 调用 时 能 够 处 理 数 据 上 传 及 数据 返 
上 加。 需要 注意 的 是 ，wsdl:part 参数 列表 必须 要 与 消 因 类 体 〈 即 后 台 PHP 类 ) 绑 定 的 方 读 参 
数 一 一 对 应 。 


12.2.6 binding 服务 绑 定 描述 


客户 站 在 调用 服务 端 接 口 时 ，SOAP RERE K WSDL 指定 的 绑 定 操作 映射 为 后 从 
PHP 中 相对 应 的 类 成 员 方 法 。 如 以 下 代码 所 示 。 


<wsdl:binding name="apiSOAP" type="tns:api"> 


<soap:binding style="rpc" transport="http://schemas.xmlsoap.org/soap/http"/> 
<wsdl:operation name="getData"> 
<soap:operation soapAction="http://www.example.org/api/getData"/> 
<wsdl:input> 
<soap:body use="]iteral"/> 
</wsdl:input> 
<wsdl:output> 
<soap:body use="literal"/> 
</wsdl:output> 
</wsdl :operation> 
</wsdl:binding> 


如 上 述 代码 所 示 ，wsdlbinding 表示 这 是 一 个 服务 绑 定 节点 ， 一 个 WSDL 文档 模型 必须 
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要 有 一 个 绑 定 描述 节点 ， 这 里 的 绑 定 名 称 为 “apiSOAP”， 对 应 该 的 类 型 为 “tns:api〈 即 
proType T RAER)”. wsdl:binding TARS 3 个 重要 的 节点 元 素 : 

> soap:binding: 表示 使 用 SOAP 通过 HTTP 协议 进行 互动 ， 并 且 以 人 RPC 进行 远程 调用 。 

> wsdl:operation: 摘 述 可 公开 的 服务 “ 即 PHP 中 的 类 成 员 方 法 ); 每 个 操作 由 两 部 分 组 

Ko TIJN input CE) M output OGRE). 

> soap:body: 描述 内 容 类 型 ， 常 用 的 值 有 use="literal" (文本 ) 和 use="encoded" 《代码 )。 

同 理 ， 如 果 需 要 公开 多 个 PHP 类 方法 为 可 调用 的 服务 ， 只 需要 在 wsdl:binding 市 点 中 添 
加 多 个 操作 即 可 。 


12.2.7 service 服务 描述 


在 WSDL 文档 模型 中 ，service 节点 始终 贯穿 整个 SOAP 请 求 流 程 。service 节点 由 多 个 


port 子 市 点 元 素 组 成 ， 一 个 port 代表 一 个 功能 类 ， 如 以 下 代码 所 示 。 
<! 一 服务 列表 1......--> 


<wsdl:service name="api"> 


<wsdl:port binding="tns:apiSOAP" name="apiSOAP1"> 
<soap:address location="http://c1.localhost/api.php"/> 
</wsdl :port> 
<wsdl:port binding="tns:apiSOAP" name="apiSOAP2"> 
<soap:address location="http://c3.localhost/api.php"/> 
</wsdl :port> 
<! 一 添加 更 多 port 
</wsdl:service> 
<! 一 服务 列表 3 
<wsdl:service name="smtp"> 
<wsdl:port binding="tns:apiSOAP" name="smtpSOAP"> 
<soap:address location="http://c1.localhost/smtp.php"/> 
</wsdl :port> 
</wsdl:service> 


如 上 述 代 人 码 所 示 ， 一 个 wsdl:port 由 binding 属性 及 name 属性 构成 ， 这 两 个 属性 都 是 可 
选 的 ， 分 别 表示 当前 服务 消 姑 类 绑 定 的 WSDL 操作 ， 以 及 当前 的 wsdl:prot 名 称 。 

一 个 WSDL 文件 可 以 同时 出 现 多 个 service， 每 一 个 service 又 可 以 同时 出 现 多 个 pont, 
这 对 于 做 分 布 式 架构 而 言 是 非常 方便 鸭 ， 例 如 根据 服务 类 型 端口 的 不 同 ， 使 用 不 同 的 service 
绑 定 文件 处 理 ， 避 免 单 一 service 绑 定 脚本 处 理 超 时 。soap:address 表示 后 台 PHP 脚本 的 URL 
地 址 ， 该 URL 地 址 必须 是 RPC 可 调用 的 URL 地 址 。 


12.2.8 可视化 创建 WSDL 


表面 次 入 介绍 了 WSDL 文件 结构 ， 相 信访 者 已 经 对 WSDL 有 了 初步 认识 。 前 面 的 内 容 
主要 是 方便 读者 深入 认识 WSDL 的 组 成 要 素 ， 以 便 在 出 错时 能 够 迅速 排除 错误 。 事 实 上 很 
多 PHP 可 祝 化 编程 工具 已 经 提供 了 全 面 的 WSDL 支持 ， 例 如 ZendStudio, Eclipse PDT 等 。 
使 用 可 视 化 工具 创建 WSDL 不 仅 可 以 提高 开发 效率 ， 还 可 以 减少 出 错 的 几率 。 接 下 来 将 以 
ZendStudio 8.0 为 例 ， 详 细 介 绍 在 IDE 环境 中 创建 WSDL 的 全 过 程 。 

首先 打开 ZendStudio， 选 中 相应 的 PHP 项 目 ， 按 下 键盘 上 的 【CtrI+N】 人 快捷 键 ， 选 择 创 
建文 件 类 型 ， 这 里 选择 “WSDL”， 如 图 12-4 所 示 。 
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单 击 “ 下 一 步 ” 按 钮 ， 确 认 文 件 保 存 路 径 及 文件 名 称 〈 需 要 确保 该 文件 能 够 被 外 部 用 户 
访问 到 )， 这 里 将 文件 命名 为 “Userwsdl”， 如 图 12-5 所 示 。 


==] = NW 
om keng DEE EE 


新 建 WSDL 文件 E 
GES WSDL 文件 | 创建 新 WSDL 文件 
| 


| 输入 或 选择 父 文件 夹 () : 
tip.localhsot 
wS): User.wsdl 


图 12-4 ”选择 文件 类 型 图 图 12-5 文件 保存 名 称 


继续 单 击 “ 下 一 步 ” 投 钮 ， 选 择 WSDL 文档 Target namespace 命名 宇 间 、xmlns:tns 命名 
HJ, JAJE “Create WSDL Skeleton” 选 项 ， 如 图 12-6 所 示 。 


Si 


Target namespace: http:nwww.example.org/Usery 
Frefie ins 


R| Create WSDL Skeleton 


@ 


图 12-6 WSDL 文件 属性 


最 后 只 需要 单 击 “ 完 成 ”按钮 ， 即 可 创建 一 个 WSDL 文件 。 现 在 的 WSDL 文件 只 有 此 
架 ， 并 没有 消息 体 。 打 开 Userwsdl 文件 后 ， 单 击 代码 编辑 区 中 的 “设计 ”标签 ， 切 换 到 可 
视 化 设计 模式 ， 如 网 12-7 所 示 。 


Eimput F parameters | ei NewOperation 
$i output F parameters | ei NewOperationResponse | 一 


图 12-7 可 视 化 编辑 WSDL 文件 


图 12-7 中 ，A 区 表示 service 服务 描述 列表 ，UserSOAP 表示 一 个 port， 如 果 需 要 添加 
多 个 port， 只 需要 选中 “UserSOAP” 服 务 列表 ， 强 出 快捷 末 单 ， 选 择 “Add Port” 命 令 即 
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可 ， 如 图 12-8 所 示 。 

这 里 只 需要 将 默认 的 port 地 址 改 为 “http://cl.localhost/user.php” 即 可 。B 区 表示 WSDL 
文档 操作 设置 区 ， 其 中 User 表示 消息 类 体 名 《〈 一 般 与 后 人 台 绑 定 的 PHP 类 同名 即 可 ); 
NewOperation 表示 一 个 操作 ， 一 个 操作 束 是 PHP 类 中 的 方法 (为 避免 混乱 最 好 保持 与 PHP 
类 成 员 方 法 同名 )， 其 中 input 表示 数据 上 传 ， 访 项 对 应 的 值 即 为 该 操作 的 传 入 参数 。output 
表示 该 操作 返回 的 数据 类 型 ， 有 效 值 为 schema 文 持 的 标准 数据 类型 。 

为 了 方便 演示 ， 这 里 将 在 User 类 中 创建 一 个 getUser 方法 ， 该 方法 文 持 两 个 参数 ， 后 台 
将 根据 传 入 的 参数 返回 该 用 户 的 数据 ，user.php 文件 代码 如 下 所 示 。 


<?php 
class User{ 


public function getUser ($username, $password) { 


if (Şusername=="admin" && Ş$password="123") { 
return "密码 正确 "; 
}elsel 


return "BIDER"; 
} 

} 
} 
ini_set ("soap.wsdl_cache_enabled", "1"); 
$server =new SoapServer('./user.wsdl',array('soap version' => SOAP_1_2)); 
Ş$server->setClass ("User"); 
if (isset ($HTTP_RAW_POST_DATA)) { 

Şrequest = ŞHTTP_RAW_POST_DATA; 
} else { 

$request = file_get_contents ('php://input'); 
} 
Şserver—->handle ($request); 
?> 


接 下 来 只 需要 绑 定 到 userwsdl 文件 即 可 。 首 先 将 默认 的 “NewOperation ”操作 改 为 
“getUser” 操 作 ， 然 后 增加 两 个 input Part《〈 即 传 入 参数 )， 如 网 12-9 所 示 。 


党 Add Operation 
la Add Fault 
EE P Add Part 
x MRO 
X MRD) Set Message » 
© Show properties 国 Show properties 
引用 (R) 引用 (R) r 


图 12-8 WIMI port 图 12-9 ”可 视 化 添加 操作 参数 


添加 的 part 默认 数据 类 型 为 string。 如 果 需 要 修改 现 有 的 part， 只 需要 选中 相应 的 part, 
在 代码 编辑 需 下 方 将 出 现 属性 编辑 窒 口 ， 如 图 12-10 rz 


E mp ite "EI rma MAR, A Debug Output vg "SP 


luserna me 


[string 


CESE Reference Kind: © spm) Zi mE 


图 12-10 元素 属 性 编辑 对 话 框 
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在 类 型 一 栏 中 ， 可 以 选择 type (WR) 定义 ， 或 者 使 用 xsd:schema 标准 数据 类 型 。 一 上 般 
选择 “类 型 ” 即 可 。 按 照相 同 的 方法 ， 为 getUser 操作 再 添加 一 个 part 并 命名 为 password。 

完成 后 ， 将 user.wsdl 与 user.php 保存 到 http://cl.localhost 网 站 根 目 录 中 。 通 过 前 面 的 步 
又 ， 一 个 SOAP 服务 文件 驶 创建 完成 了 ， 旋 者 可 以 在 客户 山中 使 用 SoapClient 进行 测试 ， 如 
以 下 代码 所 示 。 


<?php 
Sod new Soap lient neeo /Cl localhost / user wsdl); //S Pi 
Şrows=$soap->getUser ("admin", "123"); 


RS 

当然 也 可 以 使 用 第 三 方 SOAP 测试 工具 进行 测试 ， 例 如 Microsoft Visual Studio, 
soapUI、Eclipse 和 等， 上述 代 人 码 运 行 结果 为 “密码 正确 ”。ZendStudio 对 WSDL 的 文 持 还 有 很 
多 功能 ， 由 于 篇 幅 所 限 在 此 不 再 细 述 ， 读 者 可 以 一 一 动手 实验 ， 增 加 对 WSDL 的 认识 。 


12.2.9 使 用 nusoap 创建 WSDL 


nusoap 是 一 套 第 三 方 开源 类 库 ， 用 于 创建 及 调用 SOAP 服务 。nusoap 不 依赖 于 PHP H 
展 模 块 ， 在 早期 的 PHP 4.x 时 束 已 经 存在 ， 所 以 束 算 PHP 没有 开局 SOAP 模块 ， 也 不 会 影响 
nusoap 的 运行 。nusoap 非常 灵活 及 强大 ， 不 仅 文 持 钊 用 的 SOAP 服务 调用 ， 而 且 还 文 持 动态 
生成 WSDL， 文 持 使 用 数组 作为 消息 参数 次 型 。 

此 外 nusoap 套件 还 文 持 多 种 客户 问 与 服务 哨 连接 方式 ， 例 如 代理 连接 ，SSL 安全 连接 
等 。 使 用 nusoap 开发 SOAP 服务 ， 不 仅 可 以 有 效 提 高 开发 效率 ， 而 且 由 于 代码 全 部 基于 
PHP， 开 及 员 不 需要 面 对 复 杂 难 记 的 WSDL 标记 ， 只 需要 掌握 PHP AIRAA. 
下 来 将 结合 示例 代码， 详细 介绍 nusoap 的 使 用 。 

1. nusoap 入 门 

首先 下 载 nusoap， 下 载 地 址 为 http:// beauty-soft.net/book/php myvc/vendornusoap.html。 
解压 后 得 到 lib 文件 夹 及 samples XK, HP lib 文件 夹 为 nusoap 核心 文件 类 库 ; samples 
文件 夹 存 放 了 一 些 演示 示例 ， 这 里 只 需要 将 lib 文件 夹 复 制 到 网 站 nusoap 目录 下 即 可 。 接 下 
来 分 别 介绍 。 

首先 在 nusoap 目录 中 创建 一 个 服务 端 文件 并 命名 为 apiphp， 代 人 码 如 下 所 示 。 


<?php 


require_once ("./lib/nusoap.php"); 
function test ($strl, Sstr2) { 
if (is_string($strl) && is_ string($str2))t 
EE EE s SseSEr2 
}else{ 
return new soap fault (' SPD, (See '); 
} 
} 
$soap = new soap_server; 
Şsoap->configureWSDL ('api'); 
Şsoap->register ('test', 
array (Vgrcr l Y= WSC STELA y V STEZ V= SCl STELAN) E 
Ser 
) 7 
SHTTP_RAW POST DATA = isset ($HTTP_ RAW_ POST DATA) ? S$SHTTP_ RAW_ POST DATA : ''; 
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Şsoap->service (Ş$HTTP_RAW_POST_DATA); 
?> 


上 述 代码 共 定 义 了 一 个 函数 ， 并 命名 为 test。 这 里 为 了 便于 测试 ， 并 没有 实现 具体 功 
能 。 完 成 后 ， 使 用 register 方法 将 该 函数 注册 为 WSDL 操作 (operation)。register 第 2 个 参 
数 表示 message 消息 体 〈 即 operation 的 传 参 )， 第 3 个 参数 表示 操作 返回 的 数据 类 型 。 

将 文件 保存 ， 一 个 SOAP 服务 端 束 创建 完成 了 。 nusoap 强大 之 处 还 在 于 其 调试 功能 ， 
nusoap 内 置 了 一 个 简单 的 调试 器 ， 可 直接 访问 api.php 文件 ， 如 图 12-11 所 示 。 


12-11 WSDL 生成 效果 


单 击 页 面 中 的 “test” 操 作 ， 将 显示 该 操作 的 调用 方式 及 数据 类 型 ， 如 图 12-12 R. 


图 12-12 rest 操作 的 调用 说 明 
图 12-12 中 ， 详 细 地 列 出 了 test 操作 的 传递 参数 ， 返 回 类 型 以 及 调用 URL。 如 果 需 


要 查看 生成 后 的 WSDL 文件 源 代码 ， 可 以 单 击 页 面 上 的 “WSDL” 连 接 ， 戏 果 如 图 12-13 
所 示 。 
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Firefox = 
Nei http://loc...php?wsdl * |[ NuS0OAF: api x Ehttpy/loc../soap.php = | + 
é > U U G omre E- nuot A D- ~- dë - Gm 


该 XML 文件 并 未 包含 任何 关联 的 样式 信息 。 文 档 树 明示 如 下 。 | 


Fe 


- edefinitions targetNamespace="http://localhost/soap/api" > 


- <types> 
- <Xġsd:schema targetNamespace="http://localhost/soap/api" > 


<xsd:import namespace="http://schemas.xmlsoap.org 
/soap/encoding/" /> 
<xsd:import namespace="http://schemas.xmlsoap.org/wsdl/"/> 
</xsd:schema= 
</types> 
- <message name= "testRequest" > 
<Part name= "strl” type="xsd:string"/ > 
<part name= "str2" type="xsd:string"/> 
</Message> 
一 <message name= "testResponse" > 
<part name="return" type="xsd:string" /> 
GEET TE 


图 12-13 WSDL 源 代 码 


2. 调用 SOAP 服务 

nusoap 本 喘 提 供 了 用 于 调用 SOAP 服务 的 类 ， 但 是 由 于 使 用 SoapClient 作为 类 名 ， 与 
PHP WEH SoapClient 同名 ， 由 于 PHP 对 类 名 、 函 数 名 大 小 写 不 敏感 ， 所 以 如 果 局 用 了 内 
置 的 SOAP 扩展 模块 ， 束 不 要 使 用 nusoap 的 SoapClient 类 库 。 只 需要 使 用 内 置 的 SoapClient 
类 即 可 ， 如 以 下 代码 所 示 。 


<?php 


$soap = new SoapClient ('http://localhost/nusoap/ws .php?wsdl1'); 
Şrows=$soap->test ("Hello", "World"); 

var_dump ($rows); 

SE 


12.2.10 ThinkPHP 生成 SOAP 服务 


使 用 SoapServer 类 中 的 setClass 方法 可 以 将 一 个 PHP 类 中 的 所 有 方法 瀛 加 为 WSDL Hè 
作 。 根 据 这 个 原理 ， 我 们 可 以 将 一 个 模型 作为 WSDL 消 恩 体 。 这 是 MVC 开发 优势 所 在 ， 也 
是 本 书 始 终 吐 容 的 主题 。 在 MVC 中 ， 开 发 人 员 在 设计 完 一 个 模型 后 ， 后 期 的 扩展 是 非常 灵 
活 的 ， 一 个 模型 的 展现 层 可 以 是 HIML 、Json 、XML 或 者 SOAP 。 接 下 将 继续 以 
ThinkPHP 3.0 为 例 ， 详 细 介 绍 在 MVC 的 中 SOAP 服务 端 开发 ， 步 又 如 下 。 

首先 在 当前 应 用 中 创建 一 个 根 目 录 ， 并 命名 为 api， 访 目录 用 于 存放 WSDL 文件 ， 所 以 
需要 确保 用 户 能 够 访问 到 该 目录 。 按 照 前 面 介绍 过 的 方法 ， 创 建 一 个 WSDL 文件 ， 并 命名 
为 article.wsdl， 该 文件 对 应 ArticleModel 模型 。 如 图 12-14 所 示 。 


articlesOAP Fi ¥ NewDperation 


| http://www.exa... ó F NewOperationRequest E string 
F NewOperationResponse | E string 


图 12-14 article wsdl 文件 结构 图 
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接 下 来 在 home 项 目 中 创建 Article 用 户 模 型 ， 访 模型 即 为 article.wsdl 文件 的 消息 美体 。 
为 了 方便 污 示 ， 这 里 只 需要 创建 网 个 公开 方法 即 可 ， 如 以 下 代 了 但 所 示 。 
<?php 
class ArticleModel extends Model{ 
public $dataType="json"; 
/** 
* 获取 用 户 所 有 文章 
= macer description bere oso 
e 
public function getUserArticleAll (Ş$user){ 


Şrows=$this->where (array ( 
"add user"=>$user 
E E () > 
if (Şthis->dataType=="json"){ 
return json_encode ($rows); 
}else{ 


return $rows; 


} 
/** 
* 根据 文章 ID 获取 文章 标题 
* Qparam unknown type $id 
e 
public function getIdArticle ($id)t 
Şrows=$this->where (array ("id"=>Ş$id))->find(); 
Ş$title=Ş$rows[|"title"]; 
if (empty (stitle))t 
$title=" 疫 有 数据 "， 
} 


return $title; 


} 


然后 在 home 项 目 中 创建 api 控制 器 ， 该 控制 器 用 于 SOAP 服务 绑 定 ， 首 先 创建 
article.wsdl 服务 绑 定 ， 如 以 下 代 人 码 所 示 。 


<?php 


class ApiAction extends Action{ 
public function Article(){ 
DUANE Ke EE 
ini_set ("soap.wsdl_cache_enabled", "1"); 
$server =new SoapServer('./api/article.wsdl',array('soap version' => SOAP_1_2)); 
$server->setClass ("ArticleModel"); 
if (isset (S$SHTTP_ RAW POST DATA)) { 
srequest = S$HTTP_ RAW_ POST DATA: 
} else { 
Srecuest = file_get_contents ('php://input'); 
| 


Şserver—->handle ($request); 


} 


> 


此 时 通过 http://tp.localhost/index.php/api/Article 访问 Article 动作 ， 为 还 没有 在 
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article.wsdl 建立 服务 绑 定 ， 但 在 Article 动作 中 已 经 声明 绑 定 ， 所 以 出 现 错误 ， 如 图 12-15 所 


EE 


|E http//tp-local...hp/api/Article | + | 


+> p B emc m- zeng 加” ll l Em 


— <S0AP-ENV:Envelope= 
-<SOAP-ENV:Body> 
— <SOAP-ENV:Fault> 
<faultcode>WSDL=/faultcode> 
<faultstring>SOAP-ERROR: Parsing WSDL: Couldn't bind to 
service</faultstring> 
</SOAP-ENV:Fault: 
</SOAP-ENV:Body> 
</SOAP-ENV:Envelope> 


图 12-15 WSDL 绑 定 出 错 


接 下 来 需要 在 article.wsdl 文件 中 创建 相应 的 操作 及 服务 绑 定 。 这 里 将 创建 
getUserArticleAll 操作 及 getIdArticle 操作 。 并 且 建 立 http://tp.localhost/index.php/api/Article 服 
务 绑 定 地 址 。 

需要 注意 的 是 ， 在 WSDL 中 PATHINFO URL 模式 是 无 效 的 ， 所 以 上 述 绑 定 地 址 是 不 被 
SOAP 认可 的 ， 解 决 办 法 是 隐藏 index.php 入 口 文件 ， 或 者 使 用 传统 显 式 传 参 模式 〈 普 通 横 
式 ) UK REWRITE 模式 。 完 成 后 article.wsdl 设计 视图 如 图 12-16 所 示 。 


@ article 
SS getUserArticleAll 


E string 


F id 
F getldArticleResponse E string 


图 12-16 article.wsdl 文件 设计 视图 


最 终 article.wsdl 文件 的 源 代码 如 下 所 示 。 


<?xml version="1 .0" encoding="UTF-8" standalone="no"?> 
<wsdl:definitions xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/" xmlns:tns= 
"http://www.example.org/article/"xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/"xmlns :xsd= 
"http://www.w3.o0org/2001/XMLSchema"name="article"targetNamespace="http://www.example.org/ 
arceLele/“> 
<! 一 消息 --> 
<wsdl:message name="getUserArticleAllRequest"> 
<wsdl:part name="user" type="xsd:string"/> 
</wsdl:message> 
<wsdl:message name="getUserArticleAllResponse"> 
<wsdl:part name="getUserArticleAllResponse" type="xsd:string"/> 
</wsdql:messade> 
<wsdl:message name="getIdArticleRequest"> 
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<wsdl:part name="id" type="xsd:int"></wsdl:part> 
</wsdl:message> 
<wsdl:message name="getIdArticleResponse"> 

<wsdl:part name="getIdArticleResponse" type="xsd:string"></wsdl:part> 
</wsdl:message> 


<! 一 公开 可 调用 的 操作 -> 
<wsdl:portType name="article"> 
<wsdl:operation name="getUserArticleAll"> 
<wsdl:input message="tns:getUserArticleAllRequest"/> 
<wsdl:output message="tns:getUserArticleAllResponse"/> 
</wsdl:operation> 
<wsdl:operation name="getIdArticle"> 
<wsdl:input message="tns:getIdArticleRequest"></wsdl:input> 
<wsdl:output message="tns:getIdArticleResponse"></wsdl:output> 
</wsdl:operation> 
</wsdl:portType> 
<! 一 操作 绑 定 --> 
<wsdl:binding name="articleSOAP" type="tns:article"> 
<soap:binding style="rpc" transport="http://schemas.xmlsoap.org/soap/http"/> 
<!—getUserArticleAl1l 操作 --> 
<wsdl:operation name="getUserArticleAll"> 
<soap:operation soapAction="http://www.example.org/article/getUserArticleAll"/> 
<wsdl:input> 
<soap:body namespace="http://www.example.org/article/" use="literal"/> 
</wsdl:input> 
<wsdl:output> 
<soap:body namespace="http://www.example.org/article/" use="literal"/> 
</wsdl:output> 
</wsdl:operation> 


<!—getIdArticle RE 
<wsdl:operation name="getIdArticle"> 
<soap:operation soapAction="http://www.example.org/article/getIdArticle"/> 
<wsdl:input> 
<soap:body namespace="http://www.example.org/article/" use="literal"/> 
</wsdl:input> 
<wsdl:output> 
<soap:body namespace="http://www.example.org/article/" use="literal"/> 
</wsdl:output> 
</wsdl:operation> 
</wsdl:binding> 


<! 一 服务 描述 --> 
<wsdl:service name="article"> 
<wsdl:port binding="tns:articleSOAP" name="articleSOAP"> 
<soap:address location="http://tp.localhost/index.php?m=api&amp;a=Article"/> 
</wsdl :port> 
</wsdl:service> 
</wsdl:definitions> 


人 至此， 一 个 article.wsdl SOAP 服务 束 创 建 完成 了 ， 接 下 来 束 可 以 使 用 专业 的 SOAP 测试 
工具 或 者 在 PHP 中 直接 调用 该 服务 。 
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12.3 ”使 用 soapUl 测试 WSDL 


前 面 提 到 过 ， 虽 然 SoapClient 类 可 以 实现 对 SOAP 的 调用 ， 但 由 于 PHP 对 数据 类 型 不 
敏感 ， 所 以 并 不 意味 看 在 PHP 中 下 能 够 正常 调用 就 意味 看 SOAP 服务 正确 无 误 。 为 了 能 够 
让 SOAP 服务 通用 于 各 种 软件 及 便 件 平台 ， 需 要 借助 于 第 三 方 工具 进行 深入 测试 。 比 较 友好 
日 严 译 的 测试 工具 包括 soapUI、Microsoft Visual Studio, MARK Java 平台 ， 后 者 代表 微软 
设备 平台 ， 下 面 将 对 soapUI 进行 介绍 。 


12.3.1 soapUl 简介 


soapUI 是 一 人 尽 功 能 强大 、 方 便 易 用 的 第 三 方 开源 测试 工具 。 能 够 对 各 类 型 SOAP 进行 严 
紧 的 测试 ， 确 保 SOAP 适用 于 各 种 平台 。soapUI 基于 Java 构建 ， 可 以 运行 于 多 数 文 持 Java 
环境 的 果 面 系统 ， 例 如 Windows、Linux、MaxOS 等 。 

soapUI 专业 之 处 在 于 能 够 向 化 SOAP 的 测试 步 又， 以 自动 化 的 方式 为 测试 人 员 提 供 智 能 
化 的 测试 流程 ， 并 且 能 够 对 每 个 测试 单元 进行 管理 ， 文 持 对 每 个 单元 进行 赋值 测试 、 靳 言 测 
试 、 性 能 测试 等 。soapUI 测试 时 不 依赖 网 络 ， 可 以 对 本 地 WSDL 进行 测试 。 利 用 soapUI 的 
性 能 测试 工具 ， 还 能 够 对 网 站 负载 能 力 进 行 生 观 的 测试 ， 尽 早 发 现 网 站 性 能 瓶颈 之 所 在 。 
soapUI 的 项 目 网 址 为 http:/www.soapui.org/， 开 发 人 员 可 以 免费 获取 最 新 版 本 《试用 成 )。 

soapUI 文 持 以 独立 的 软件 包 进 行 安 朔 ， 也 文 持 以 插件 的 方式 整合 到 Eclipse 
maven2.X、Netbeans 、intellij 等 IDE 环境 中 。 界 面 如 图 12-17 所 示 。 


S E articleS0AP 
Si getUserArticleAll 
ZZ Request 1 
-S getkdArticle 
Za 


Sg 
IG E articlesOAP Test5uite 


soapUl log httplog jetty log errorlog wsrmlog memory log tools 


图 12-17 soapUI 界面 


如 图 12-17 所 示 ，soapUI 工具 界面 大 致 可 分 为 四 大 部 分 ， 分 询 如 下 : 
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> A: 表示 项 目 管理 区 。 通 常情 况 下 ， 每 一 个 项 目 对 应 一 个 WSDL 文件 。 

> B: 表示 操作 请 求 区 。 测 试 人 员 双 击 项 目 管理 区 的 操作 返回 节点 ， 操 作 请 求 区 将 要 求 
输入 请 求 参 数 。 

> C: 表示 请 求 结果 区 。 显 示 请 求 的 结果 。 

> D: 工具 栏 。 图 标 表示 开始 执行 请 求 ， 蜂 图 标 表示 增加 断言 测试 。 


12.3.2 ”安装 soapUl 


接 下 来 将 使 用 soapUI 独立 安装 包 对 前 面 创建 的 article.wsdl 文件 进行 测试 。 首 先前 往 官 
方 网 站 下 载 相 应 的 Java 源 代 人 码 包 ， 这 里 下 载 的 版 本 为 soapUI 4.$.1。 下 载 完成 后 解压 将 得 到 
soapUI 4.5.1 目录 ， 该 目录 存放 了 soapUI 所 需要 的 源 代码 文件 。 在 启动 soapUI 工具 之 前 ， 需 
要 确保 当前 机 器 上 已 经 安装 Java 环境 。 笔 者 的 环境 如 图 12-18 所 示 。 


San 命 他 提示 符 


Hicrosoft Windows [RÆ 6.1.7601] TES | 
RAPS (ei 2003 Hicrosoft Corporation. F E PTA AUA. 


C: \Users\Administrator>java -vērsi 
31" 


java version “1.6.0_ 
Java{ TM) SE Runtime Environment (build 1.6.0_31-b05) 
Java Hotspot(TH) Client UH {build 20.6-b01, mixed mode, sharing) 


C: \Users\Administrator >? 


图 12-18 Java 环境 


EMUR Java 环境 后 ， 只 需要 进入 soapUI 4.5.1 Ha, 然后 再 进入 bin 目录 ， 双 击 
“soapui.bat” 批 处 理 文 件 ， 此 时 操作 系统 会 询问 是 否 允 许 soapUI 通过 防火 墙 ， 选择 “允许” 
按钮 即 可 。 首 次 局 动 界 面 如 图 12-19 所 示 。 


3 soapUl 4.5.1 


Die Tock Desktop Help 
3aogpaCGo ën SE 


soapUl vs. gien PRO 


- A Rumble in the the ae Jungle 


FREE! 
Read more | Download Trial 


soap ilog tp log jettylog emorlog wsmiog memory iog 


图 12-19 soapUI 欢迎 界面 
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12.3.3 创建 项 目 


ah soapUI 后 ， 首 先 创建 一 个 测试 项 目 。 单 击 “ 项 目 管 理 区 ” 然后 按 下 鼠标 右键 ， 弹 
HEREZE, wF% “New soapUI Project” 命 令 ， 如 图 12-20 所 示 。 


| 站 soapUl 4.5.1 


Die Iools Desktop Help 
langs sp Lt 


-一 hanl Starter Da, pr 
- 
Jew 5 Droe ct Cri-N 
LÉI : KS 一 
part Droe i- 
: "e ME Oje | i P Ce Vs. 
TT 
s LG 


Import Remote Project 
e in the Testing Jungle 
Save All Projects Ti+ AR-5 


Open All Closed Projects fg 
Close All Open Projects | 


Rename 

New Workspace 
Switch Workspace 
Clear Workspace 


Online Help 


Download Faal 


soaplllog httplog jettylog erorlog warm log memory log | 


图 12-20 创建 测试 项 目 


问 导 将 进入 “New SoapUI Project” XEHE, E Project Name 选项 栏 中 输入 测试 项 目 名 
称 ， 这 里 命名 为 article; Initial WSDL/WADL 文本 框 中 是 WSDL 文件 地 址 ， 可 以 选择 本 地 或 
者 远程 URL。 如 采 选 择 本 地 文件 ， 可 以 单 击 “Borwse” 投 钮 进行 选择 ， 这 里 只 需要 输入 
“http://tp.localhost/api/article.wsdl” 地 址 即 可 ;完成 后 如 图 12-21 所 示 。 


五 New soapUI project 


New soapft Project 
Creates a new soapUl Project in this workspace 
‘| Project Name: 
| Initial WSDL/WADL: 
Create Requests: Create sample requests for all operations? 


Create TestSuite: Creates a TestSuite for the imported WSDL or WADL 
Create MockService: i 


Add REST Service: [_] Opens dialog to create REST Service 
Relative Paths: Stores all file paths in project relatively to project file (requires save) 


图 12-21 测试 文件 地 址 
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单 击 “OK” 按 钮 ， 回 导 将 进入 文件 保存 对 话 框 ， 默 认 情 况 下 soapUI 使 用 “project 名 称 
+- soapui-project.xml ”的 形式 进行 命 。 所 以 article 项 目的 文件 名 称 为 “article-soapui- 
project.xml 。 


单 击 “ 保 存 ” 按 钮 后 ， 同 导 将 进入 “Generate TestSuite” 对 话 框 ， 在 Operations 选项 栏 


@ One TestCase for each Operation 
O Single TestCase with one Request for each Operation 
© Use existing Requests in Interface 
@ Create new empty requests 
Operations: wl getidArticle 
getUserArticleAll 


Select all Unselect all 


| Generate LoadTest: enerates a default LoadTest for each created TestCase 


图 12-22 soap 生成 测试 选择 框 


Hi “OK” H, AMRAH Hm AHE, wE 12-23 Mar. Eii “ME” i 
SUD). JERN Hem A or or DI GT DEI DD, UAM HSF E. OR 12-24 
所 示 。 


Projects 

o E article 

8 G-I articleSOAP 

e TestSuite 由 -党 getldArticle 


Lë Enter name of TestSuite to create | : H-S getUserArticleAll 
articlesOAP TestSuite| 


G E articleSOAP TestSuite 
-g getIdArticle TestCase 


SG sg getUserArticleAll TestCase 
图 12-23 ”测试 用 例 命 名 对 话 框 图 12-24 项 目 管理 区 


12.3.4 MNAR H 


通过 前 面 的 步骤 ， 已 经 增加 了 article 测试 项 目 ， 并 日 soapUI 已 经 根据 WSDL 的 
Schema 定义 为 每 一 个 操作 创建 了 默认 请 求 。 接 下 来 首先 测试 getdArticle 操作 。 

依次 单 击 “articleSOAP TestSuite ”一 “getIdArticle TestCase ”市 点 ， 然 后 双击 
“getIdArticle” 贡 点 ， 弹 出 测试 请 求 窗口 ， 如 图 12-25 所 示 。 
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Ta0apenv Envekipa> 


, Head... Attachm.. W. WS.. JMS Ha. JMS Prop.. 


© Assertions (0) Request Log (0) 


图 12-25 测试 请 求 窗口 


代码 中 的 id 元 素 表 示 请 求 参数 名 称 ,“?” 表 示 需 要 传递 的 参数 值 。 输 入 相应 的 参数 值 ， 
单 击 工具 栏 W 控 钮 ， 右 边 的 请 求 结 来 区 将 显示 SOAP 返回 的 结 来 ， 如 图 12-26 所 示 。 


„J... | Headers (12) Attachments (0) si Info WSS (0) JMS (0) 


© Assertions (2) Request Log (2) 
response time: 129ms (377 bytes) C1:1 


图 12-26 SOAP 返回 结 
利用 同样 的 方法 ， 对 其 他 公开 的 操作 进行 测试 ， 这 就 是 soapUI 的 简单 使 用 。 如 果 


WSDL 出 现 异 常 ， 请 求 结果 窗口 将 返回 详细 的 异常 说 明 ， 开 发 人 员 可 以 根据 这 些 说 明 查找 异 
常 所 在 。 


12.3.5 ”负载 测试 
由 于 SOAP 本 质 上 是 一 种 XML 通信 ，XML 在 进行 数据 传递 时 ， 需 要 传输 大 量 的 标记 
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元 系 ， 这 对 接口 的 性 能 是 一 个 挑战 ， 所 以 性 能 测试 也 是 衡量 一 个 SOAP 服务 是 否 稳定 的 关 
键 。 网 站 性 能 测试 ， 一 般 使 用 loadrunner 工具 进行 测试 ， 广 工具 是 一 套 专 业 且 强大 的 商业 测 
试 工具 ， 使 用 loadrunne 能 够 对 网 站 进行 全 面 的 性 能 测试 ， 并 以 报表 的 方式 返回 结果 ， 为 开 
发 人 员 优 化 程序 提供 可 靠 的 数据 参考 。 但 loadrunner 使 用 烦琐 ， 对 于 一 般 程序 员 而 言 ， 上 手 
是 一 件 困难 的 事 。 接 下 来 将 介绍 soapUI 内 置 的 负载 能 力 测试 工具 ， 该 工具 不 仅 使 用 简单 ， 
而 且 界 面 友好 ， 完 全 能 够 满足 一 般 的 SOAP 性 能 测试 需求 。 

首先 在 项 目 管理 区 中 选择 “article” 项 目 ， 然 后 依次 单 击 “articleSOAP TestSuite ”一 
“Load Tests” 节点 。 双 击 “LoadTest 1” 节 上 点， 弹出 LoadTest 1 测试 对 话 框 ， 如 图 12-27 所 示 。 


YEIERFC Ee 
Threads: 1008} Strategy imole =) Test Delay Ca Random 
) Test Step | eege , avg | e gege ps Ges wurde Se 


getldArticle 
TestCase: 0 i S ; : S S = i 0 
A7 Show Types: äi EE Show steps: 


二 HL 三 Je La _ 
t 2012-11-25 14:34... Message LoadTest tartod: at Se Nov 25 14:. 


1 2012-11-25 14:35... Message LoadTest ended at Sun Now 25 14: a. 


0 entries 
LoadTest Leg LoadTest Assertions Setup Script TearDown Script 


图 12-27 soapUI 性 能 测试 对 话 框 


如 图 12-27 所 示 ， 可 设置 的 测试 选项 如 下 : 

> Limit: 测试 超时 规则 ， 单 位 有 Total Runs、Seconds、Runs pre Threads。 

> Threads: 局 动 的 测试 线程 〈《 即 访问 用 户 )。 

> Strategy: 测试 荣 略 ， 可 选 的 值 有 Burst、Simple、Threads、Varinace。 

测试 对 话 框 大 致 可 分 为 3 部 分 : 顶部 的 为 测试 设置 栏 ， 中 间 区 域 表 示 测 试 状 态 显 示 区 ; 
底部 为 测试 结果 显示 区 。 这 里 将 Threads 值 设 为 100; Limit 值 设 为 600 Seconds- 

完成 后 单 击 # 按 钮 ， 执 行 测 试 。 根 据 网 站 性 能 及 网 络 状态 ， 将 会 耗 时 一 段 时 间 ， 完 成 后 
soapUI 将 返回 相应 的 测试 数据 ， 如 图 12-28 所 示 。 


bX 加 “加 0 Limit: | ez [Seconds ` ~] [ 100% ] 
Threads: 1 10015 Strategy [imole ~] Test Delen | O] Random [| 05| 
Test Step min | max avg last cnt tps | bytes| bps err rat 
TestCase: 2023 13020 5,816... 5901 10261 17.07 38683.. 6437 0 0 
ix a Show Types: [- All - -| Show Steps: 


图 12-28 ”性 能 测试 结 
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如 图 12-28 所 示 ，SOAP 服务 接口 经 过 测试 后 ，soapUI 将 以 直观 的 数据 反馈 给 开发 人 
员 。max 数值 越 小 ， 证 明 性 能 越 好 。 点 击 工具 栏 中 的 轿 按 钮 ， 将 以 曲线 图 的 方式 显示 结果 数 
据 ， 如 图 12-29 所 示 。 


ThreadCount W Average (ms) ErorCount W Transaction/Sec E Bytes/Se 


图 12-29 以 曲线 图 显示 性 能 测试 结果 


12.4 ”小结 


本 章 首 先 讲 解 了 分 布 式 开发 概念 ， 然 后 介绍 了 PHP 对 SOAP 文 持 的 情况 ， 结 合 人 简单 明 
了 的 示意 图 ， 相 信 读 者 已 经 对 SOAP 有 所 认识 。 

接 看 深入 介绍 WSDL HEAL, AAi types、portType、message、binding 及 
service 。 只 有 深入 理解 这 些 文档 节点 或 元 素 的 意义 ， 才 能 设计 出 功能 强大 ， 运 行 稳定 的 
SOAP 服务 。 

本 章 还 介绍 了 nusoap KE, rk Lie Cat nl Dech tu SOAP 服务 ， 是 PHP 
平台 中 比较 好 用 、 强 大 的 SOAP FRERE. 

在 PHP 中 要 测试 SOAP 是 比较 困难 的 事 ， 本 章 最 后 介绍 了 soapUI 开源 工具 ， 使 用 该 工 
具 可 以 对 任何 类 型 的 SOAP 服务 进行 全 面 的 测试 。 
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内 容 提 要 


Smarty 是 PHP 中 一 套 非常 完善 、 强 大 的 PHP 模板 引擎 ， 真 正 实现 了 PHP 代码 与 界面 
HTML 代码 相 分 离 。 尽 管 PHP 技术 已 经 发 展 得 非常 完善 ， 各 种 MVC 框架 的 出 现实 现 了 
Smarty 代码 分 离 的 功能 ， 但 Smarty 凭借 着 稳定 的 性 能 、 高 效 的 模板 标签 、 只 活 的 扩展 机 制 
使 其 仍然 具有 不 可 替代 的 作用 。 

正 因 如 此 ， 越 来 越 多 的 主流 MVC 框架 开始 支持 使 用 Smarty 作为 模板 引擎 ， 例 如 
CakePHP, ThinkPHP 等 ， 这 些 MVC 框架 均 可 以 通过 扩展 或 内 置 的 视图 引擎 开关 引入 
Smarty 引擎 。 由 于 本 书后 面 的 实战 部 分 需要 使 用 Smarty 的 较 多 功能 ， 所 以 本 章 将 会 深入 
介绍 Smarty 的 方方面面 ， 读 者 只 有 全 面 掌握 本 章 内 容 ， 才 能 对 实战 部 分 内 容 进 行 学 习 。 


学 习 目 标 


了 解 Smarty 的 特点 。 

掌握 在 ThinkPHP 中 整合 Smarty 的 方法 。 
掌握 Smarty 利用 标签 的 使 用 方法 。 
理解 并 熟悉 Smarty 变量 调节 峰 。 
理解 并 熟悉 Smarty 对 象 方法 。 

掌握 Smarty 插件 扩展 的 使 用 。 

掌握 Smarty 强大 的 绥 存 功能 。 
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13.1 Smarty 模板 引擎 介绍 


Smarty 并 不 是 一 项 新 的 PHP 技术 ， 而 在 PHP 4.x 以 前 束 已 经 是 非常 流行 的 界面 与 后 从 
相 分 离 的 技术 。 由 于 PHP 的 特性 ， 早 期 的 PHP 编程 要 实现 MVC 编程 非常 困难 ， 但 Smarty 
出 现 之 后 ， 虽 不 能 说 是 MVC 编程 ， 但 人 至少 实 现 了 部 分 MVC 编程 思想 ， 以 全 于 现在 主流 的 
MVC 框架 或 多 或 少 都 受 Smarty 影 啊 。 

Smarty 的 本 质 是 分 离 代 码 ， 使 得 美工 与 后 台 的 旬 辑 在 一 定 程 序 度 相 分 离 ， 以 便 更 好 地 分 
工 合作 。 这 种 分 工 合作 不 仅 能 够 提高 开发 效率 ， 而 且 能 够 有 效 改善 代 但 质量 ， 为 后 期 网 站 的 
维护 提供 极 大 的 便利 ， 如 图 13-1 所 示 。 

在 早期 的 PHP 开发 模式 中 ， 能 够 真正 实现 MVC 设计 的 PHP 框架 并 不 多 ， 所 以 当时 几 
乎 由 Smarty EF. HEME, Smarty 依然 盛行 ， 这 点 从 各 大 招聘 网 站 中 可 见 一 班 。 本 书 是 
介绍 PHP MVC 的 ， 理 论 上 MVC 框架 束 能 实现 美工 与 界面 相 分 离 ， 但 MVC 是 一 种 模式 ， 
而 Smarty 只 是 模式 中 的 一 个 部 件 而 已 。 所 以 要 深入 理解 这 两 者 之 间 的 关系 ， 还 需要 读者 中 
正 掌握 Smarty 的 使 用 。 

Smarty 与 MVC 不 同 之 处 在 于 Smarty 并 不 负责 后 台所 和 辑 的 处 理 ， 普 通 的 PHP 程序 只 需 
要 包含 Smarty 入 口 文件 ， 其 他 的 编程 模式 可 以 使 用 传统 的 PHP 面 癌 对 象 或 面 癌 过 程 方式 实 
现 。Smarty 在 处 理 模 板 时 ， 首 先 由 PHP 分 析 当 前 请 求 是 否 应 用 Smarty， 然 后 对 相关 的 请 求 
文件 进行 静态 化 处 理 ， 如 果 存 在 绥 存 则 直接 返回 访 文 件 的 缓存 ;否则 执行 模板 解释 工作 ， 将 
前 人 台 设 计 人 员 设 置 的 模板 标签 解释 为 PHP 标准 代码 ， 并 且 生 成 预 编 译文 件 。 

这 个 过 程 只 需要 在 第 一 次 解释 模板 或 者 模板 标签 改变 时 执行 ， 下 次 执行 时 直接 获取 解释 
后 的 结果 即 可 ， 避 免 重 复 解 释 ， 这 在 一 定 程度 上 改善 了 PHP 运行 效率 ; 最 后 将 生成 后 的 绥 
存 返 回 给 浏览 用 户 ， 如 网 13-2 所 示 。 


Smarty 模 板 引 擎 


如 

deg eg Eë 
界面 设计 人 员 不 
是 

第 


WML XML 


父 旺 f 


读 取 并 解释 模板 


编译 后 的 PHP 


解释 模板 ， 并 返回 结果 


wi 
~ 


图 13-1 Smarty 前 人 台 后 分 离 图 13-2 Smarty 执行 流程 


国 国 355 


PHP MVC 开发 实战 


相信 读者 对 Smarty 的 执行 流程 并 不 阳 生 ， 因 为 前 面 介 绍 的 ThinkPHP 模板 引擎 束 是 以 相 
同 的 方式 处 理 模板 的 。 不 仅 同样 文 持 PHP 编译 ， 而 且 使 用 assign 方法 分 配 变 量 ; 在 模板 设 
计 层 面 ， 同 样 也 使 用 标签 ， 这 些 都 是 Smarty 原本 束 有 的 特色 功能 。 

除了 Smarty 引擎 之 外 ， 第 见 的 模板 引擎 如 下 。 

> PHPLIB: 一 套 古 老 且 主流 的 模板 引擎 ， 直 接 在 HTML 中 使 用 PHP 变量 进行 编程 。 

> Template Blocks: 一 束 轻 巧 且 速度 非常 快 的 PHP 模板 引擎 ， 文 持 XML 语法 。 

> TinyButStrong: 小 强 模板 ， 业 界 非 党 车 名 好 用 的 模板 引擎 ， 和 直接 文 持 Dreamweaver 

插件 编辑 。 

> Ram TPL: 多 于 使 用 和 安装 引擎 ， 有 6 个 标签 、3 个 PHP 函数 和 2 个 PHP 类 。 文 持 

对 模板 中 的 相对 路 径 目 动 转换 为 绝对 路 径 。 
> PHPTAL: PHPTAL 是 一 个 ZPT 的 PHP 执行 。 简 而 言 之 ，PHPTAL 是 一 个 PHP 下 的 
XML/XHTML 模板 库 。 

> PHP Template Engine: 类 似 于 PHPLIB， 但 支持 在 模板 中 使 用 Cookie, Session, 

通过 前 面 的 介绍 ， 可 以 看 到 PHP 模板 引擎 主要 分 为 两 类 : 一 类 是 直接 在 模板 中 使 用 
PHP 语法 作为 模板 标签 ， 男 外 一 类 是 使 用 特定 的 语法 标记 作为 模板 标签 。 在 这 两 类 模板 中 ， 
基于 PHP 语法 的 模板 引擎 由 于 不 需要 开发 员 掌 握 额 外 的 知识 ， 所 以 越 来 越 受 到 开发 人 员 的 
欢迎 ， 本 书 一 开始 介绍 的 几 种 主流 MVC 框架 模板 引擎 都 使 用 类 似 技术 。 

使 用 特定 语法 的 模板 引擎 由 于 性 能 问题 ， 近 年 来 受到 一 些 诉 病 ， 但 由 于 其 能 够 真正 将 界 
面 设计 人 员 与 PHP 开发 人 员 很 好 地 分 离 ， 所 以 在 大 型 项 目 开发 中 具有 不 可 百代 的 作用 ， 而 
这 一 类 的 模板 引擎 最 流行 的 就 是 Smarty 了 。 上 所 以 本 重重 点 介绍 Smarty 3 引擎， 读者 在 深入 竺 
握 该 模板 引擎 的 使 用 后 ， 将 会 对 MVC 设计 有 更 深入 的 认识 。 


13.2 使 用 Smarty 


Smarty 的 兼容 性 非常 好 ， 多 数 情 况 下 Smarty 高 版 本 都 能 够 兼容 低 版 本 。 但 有 一 点 必须 
要 注意 ，Smarty 之 所 以 功能 强大 ， 是 由 于 文 持 众多 模板 插件 ， 这 些 插件 并 非 都 具备 民 好 兼容 
性 ， 所 以 在 使 用 前 或 升级 前 需要 开发 人 员 上 自行 测试 ， 以 确保 适用 新 的 Smarty 版 本 。 接 下 来 
将 以 Smarty 3.1.12 为 例 ， 详 细 介 绍 Smarty 的 安 猴 及 使 用 。 


13.2.1 在 PHP 中 使 用 Smarty 


在 传统 的 PHP 中 使 用 Smarty 是 主要 的 开发 方式 ， 也 是 最 价 单 的 使 用 方式 。 下 面 将 通过 
示例 详细 介绍 Smarty 在 PHP 中 的 导入 及 使 用 ， 步 骤 如 下 。 

首先 下 载 Smarty， 下 载 地 址 为 http://beauty-soft.net/book/php_mvc/down/Smarty.html。 将 
下 载 后 的 Smarty 文件 夹 复制 到 网 站 相应 的 目录 ， 例 如 Smarty 目录 。 完 成 后 可 以 通过 
http://localhost/smarty 网 址 访问 到 该 目录 。 然 后 在 该 目录 中 分 别 创建 cache, compiles, conf, 
ml 目录 ， 用 于 存放 绥 存 、 编 译文 件 、 配 置 文件 、 模 板 文件 等 。 

完成 后 ， 创 建 index.php 网 站 入 口 文件 ， 在 该 文件 中 包含 Smarty.class.php 入 口 文件 ， 并 
初始 化 相关 配置 信息 ， 如 以 下 代码 所 示 。 
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<?php 
include_once ("./smarty/Smarty.class.php"); 


Şsmarty=new Smarty ();}; 


// 模 板 目录 

Ma > em le on 

/ / WHK 

$smarty->compile dir='./compiles'; 
Ee 

sil ely EOL ee Con, 

// 缓 存 存 放 目 录 
$smarty->cache_dir='./Cache'; 


Şsmarty->caching=true; 

// 模 板 定 界 符 
Şsmarty->left_delimiter='{'; 
Şsmarty->right_delimiter='}"'; 
Şrows=array (0=>array ( 

"title"=>" 标 题 一 "， 

"ncontent"=>" 内 容 "， 

), 

l1=>array ( 

meirle = np 

"content "PA, 

) v 
Şsmarty->assign("rows", SowsS ) ; 
$smarty->assign ("pageTitle", "第 一 个 Smarty 程序 "); 
Şsmarty->display ("index.html"); 


通过 前 面 章节 的 学 习 ， 相 信 读 者 对 上 述 代码 并 不 陌生 。 无 论 是 分 配 变 量 还 是 分 配 模板 ， 
这 些 操作 都 与 ThinkPHP 相 类 似 ， 事 实 上 这 是 类 Smarty 模板 引擎 所 具有 的 特性 。 保 存 上 述 代 
但 ， 然 后 在 tpl 目录 中 创建 index.html 模板 文件 ， 代 人 码 如 下 。 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" “http://www.w3.org/ 


TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 
<html xmlns="http://www.w3.o0org/1999/xhtm1"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
<title>{$pageTitle}</title> 
</head> 
<body> 
{foreach from=$rows item=list} a mn: 
<li>{$list["title"]}</li> al el httpynocalh.. D - D 
{/foreach)} pm l E 


</body> 

</html> 

如 上 述 代码 所 示 ， 标 粗 的 即 为 后 台 PHP ÆR. er 
结 示 如 图 13-3 所 不 。 图 13-3 Smarty 运行 结果 


E En 
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13.2.2 开局 ThinkPHP 模板 扩展 


前 面 介绍 的 Smarty 使 用 方式 是 基于 传统 PHP 开发 的 ， 并 没有 涉及 设计 模式 。 事 实 上 
ThinkPHP 已 经 内 置 Smarty 引擎 文 持 ， 并 且 直 接 文 持 在 配置 文件 中 开启 ， 过 程 如 下 。 

首先 打开 MVC 项 目 配置 文件 ， 将 TMPL_ENGINE_TYPE 配置 项 的 值 修改 为 smarty 即 
可 ， 如 以 下 代码 所 示 。 


<?php 


return array 人 
'TMPL_ENGINE_TYPE' => 'smarty', 
VORMIS Iy IAMS IAI SUME = V NEL y 
'TMPL ENGINE CONFIG' => array ( 
'template_dir' =>APP_PATH. 'Tp1/', 
cache dir! => APE PATH. 'Runtime/'.'Cache!, 
'config_dir' => APP_PATH.'Conf/', 
'compile dir' => APP_PATH. 'Runtime/'.'compiles', 
'compile_check' =>true, 
'use_sub dirs' => true, 
'caching'=>false, 
SEH ; 


Veigar celimicter=>V==>t, 


1: 


?> 


如 上 述 代 码 所 示 ， 除 了 TMPL_ENGINE_TYPE 配置 项 外 ， 第 三 方 模板 引擎 配置 参数 需 
要 赋值 于 TMPL_ENGINE_CONFIG 配置 项 (配置 值 与 在 传统 的 PHP 中 保持 一 致 )。 如 果 
TMPL_ENGINE_CONFIG 为 空 ， 则 使 用 ThinkPHP 内 置 引 擎 的 相关 参数 代替 。 切 换 模 板 引 擎 
后 ， 其 他 的 变量 分 配 和 默认 引擎 保持 一 致 ， 但 模板 标签 的 使 用 必须 遵 衢 Smarty 标准 及 规 
w, WA PIRR. 


public function index (){ 
Sthis->assign('pageTit1le' "网 页 标题 ") ; 
EE EE ee 


} 


13.2.3 ”以 扩展 的 方式 使 用 全 功能 Smarty 


在 11.1.2 市 中 ， 已 经 介绍 过 ThinkPHP 虽然 可 以 切换 到 Smarty 模板 引擎 ， 但 这 种 切换 方 
式 是 指标 签 解释 的 方式 上 ， 并 不 包括 Smarty 模板 扩展 功能 。 所 以 使 用 TMPL_ENGINE_ 
TYPE 切换 方式 ， 只 能 使 用 Smarty 的 基础 解释 功能 ， 而 不 能 使 用 Smarty 特有 的 扩展 功能 ， 
例如 注册 函数 、 注 册 对 象 等 。 而 Smarty 灵活 及 强大 之 极 就 在 于 扩展 机 制 。 但 可 以 通过 
ThinkPHP 扩展 的 方式 引入 Smarty， 从 而 实现 全 功能 的 Smarty 模板 引擎 。 具 体 方法 读者 可 参 
Se erg 

配置 完成 后 ， 只 需要 在 项 目 conf 目录 中 创建 confing.conf 配置 文件 即 可 ， 该 文件 用 于 直 
接 在 模板 中 获取 配置 数据 。 这 样 ThinkPHP 就 具备 全 功能 的 Smarty 环境 了 。 

本 章 接 下 来 的 全 部 内 容 基 于 ThinkPHP+Smarty 环境 。 继 续 以 mobile 项 目 作 为 演示 用 
例 ， 此 时 通过 http://tp.localhost/mobile.php 网 址 将 能 够 访问 到 该 项 目 。index 动作 代 人 码 如 下 
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所 示 。 


<?php 

// 本 类 由 系统 目 动 生成 ， 仅 供 测 试用 途 

class IndexAction extends PublicAction { 

public function index (){ 

Şthis->smarty->template_dir=APP_PATH.'/Tpl/index'; 
Sthis->smarty->assign('pageTit1le' "欢迎 光临 ") ; 
人 H:i:s")); 
sarticle=M ("Article"); 
Şrows=$article->where (array ("add_user"=>"ceiba"))->select(); 
Şthis->smarty->assign("list",$Şrows); 
Şthis->smarty->display ('index.html'); 


} 


E> 


可 以 看 到 index 控制 器 不 再 继承 于 Action 类 ， 而 是 继续 于 PublicAction 类 。 该 类 是 一 个 
目 定 义 的 项 目 成 员 控 制 器 ， 用 于 全 局 控制 ， 代 人 码 如 下 。 


<?php 


class PublicAction extends Action{ 

public $smarty; 

// 初 始 化 Smarty 

public function _initialize() { 
// 加 载 Smarty 模板 扩展 
vendor (VSnarty -ouma TYN, EE 
Şthis->smarty=new Smarty () ， 
// 默 认 模板 目录 
Şthis->smarty->template_dir=APP_PATH.'Tp1/'; 
// 编 详 目 录 
Şthis->smarty->compile_dir=APP_PATH. 'Runtime/'.'compiles' 
人 
Şthis->smarty->config_dir=APP_PATH.'Conf/'; 
// RHK 
Şthis->smarty->cache_dir=APP_PATH. 'Runtime/'.'Cache' 
Şthis->smarty->caching=false; 
EE Celimicer="<l1=={ Y; 


GEET Celimirter=" }==>" 


} 


2> 


这 里 的 Smarty 引擎 是 ThinkPHP 提供 的 ， 如 果 读 者 需要 最 新 版 本 的 Smarty, He ë8 e 
mi Smarty 目录 即 可 。 也 可 将 Smarty 存放 于 当前 项 目 扩展 目录 ， 然 后 使 用 import 函数 导入 即 
可 。index 动作 对 应 的 index.html 模板 文件 如 以 下 代码 所 示 。 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www. w3. 
org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> 

<html xmlns="http://www.w3.o0org/1999/xhtm1"> 

<head> 


<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 

<meta name="viewport" content="width=device-width, initial-scale=1"> 

<link rel="stylesheet" type="text/css" href="http://beauty-soft.net/js/jquery. 
mobile-1.0/jquery.mobile-1.0.min.css" /> 
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<script src="http://code.jquery .com/jquery-1.4.3.min.js"></script> 

<script src="http://code.jquery.com/mobile/1.0al/jquery.mobile-1.0al.min. js"> 
</script> 

<title><!--{$pageTitle}--></title> 

</head> 

<body> 

SE 

<div data-role="page" id="Index_Page" class="type-index"> 

<dir data-role="header" data-position="fixed"><h1><!-—-{$pageTitle}--></h1></dir> 

<!-- HN SS -> 

<div data-role="content"> 

<ul data-dividertheme="b" data-theme="c" data-role="]l]istview" data-inset="true"> 


<l—-{foreach from=$list item=rows } 一 一 > 
SE href="; EE EE ee GE 
<!—-{/foreach}--> 
</ul> 
</div> 
SE 


e E Ge E Ee E EE Bt SE ME 
<h4><!-—-{$foot }--></h4> 

SE 

</div> 

<!-- 首 页 END 

</body> 

</html> 


上 述 代 人 码 是 一 个 基于 JqueryMobile 的 手机 页 面 代 码 ， 并 且 使 用 Smarty 标签 及 语法 。 访 
者 在 实验 时 可 以 使 用 普通 的 模板 代码 ， 这 里 只 是 为 了 减少 代码 量 ， 方 便 讲解 ， 对 Smarty 的 
功能 介绍 并 无 有 影响。 运行 效果 如 图 13-4 PTR. 


库 克 就 苹果 地 图 缺陷 向 用 户 道 散 : 未 提供 一 流体 验 
索尼 投资 6.44 亿 美元 成 为 奥 林 巴 斯 第 一 大 股东 
| 夏普 称 会 制造 足够 Phone5 所 需 的 显示 屏 


© 
© 
© 
TRUR SIEA ER : SEATA IHARA OD 
机 构 调 查 称 Facebook 中 国 用 户 数 超 6000 万 心 

© 


| 奉 雇 寄 称 仍 末 托 到 适合 移动 瑟 联 网 的 商业 模式 


现在 时 间 2012-11-27 15:30:14 


图 13-4 E ThinkPHP 中 使 用 Smarty 


13.3 Smarty 模板 函数 和 标签 


Smarty 模板 引擎 内 置 了 很 多 函数 ， 这 些 函 效 只 允许 开 有 人员 调 用 ， 但 并 不 允许 修改 。 接 
下 来 将 介绍 弟 用 和 重要 的 内 置 函 数 。 
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13.3.1 include 〈 包 含 文件 ) 


include 函数 是 一 个 常用 的 函数 ， 与 其 他 语法 或 函数 一 样 ，include 函数 也 是 用 于 包含 站 
内 文件 的 。Smarty 引擎 中 的 include 函数 文 持 包含 模板 文件 、 资 源 文 件 〈 如 图 片 、CSS、JS 
等 )。 合 理 使 用 include 还 可 以 实现 网 站 布局 功能 ， 如 以 下 代码 所 示 。 


<!--head 区 域 --> 

<!—-{include file="mobile_head.html1"}--> 

<!-- 内 容 区 -=-> 

<div data-role="content"> 

<ul data-dividertheme="b" data-theme="c" data-role="listview" data-inset="true"> 


<!—-{foreach from=$list item=rows}--> 
<li><a href="#UserPage" ><!—-{Ş$Şrows["title"]}--></a></li> 
<!——{/foreach}——> 
< > 
</div> 


有 
<1I———{1neluees £1le= E El ene 
</html> 


上 述 代 码 将 原本 元 长 的 代码 进行 了 简化 ， 使 用 include Atara EA mobile bead bm 及 
foot.html 文件 ， 这 两 个 文件 作为 网 页 公共 头 部 及 底部 ， 实 现 了 简单 的 布局 功能 。 需 要 注意 的 
是 ，include 导入 路 径 是 相应 于 当前 模板 文件 位 置 的 〈 文 持 绝对 路 径 方式 )， 上 述 代码 中 衫 包 
含 的 文件 需要 与 index.html 文件 同 级 。 此 外 ，include 还 文 持 参数 传递 ， 如 以 下 代 但 所 示 。 


<!—-{include file="foot.html" copyright="copyright beauty-soft.net"}--> 


foot.html 负面 在 接收 传 参 时 ， 直 接 填 写 变 量 名 称 即 可 。 如 果 存 在 相同 变量 名 ， 该 变量 值 
将 和 被 丛 换 ， 如 以 下 代码 所 示 。 


EE 


13.3.2 capture 〈 暂 存 数 据 ) 


capture 是 一 个 模板 数据 暂 存 标签 ， 一 般配 合 include 函数 使 用 。 通 常情 况 下 ， 使 用 
include 包含 文件 时 ， 相 应 的 代码 会 被 立即 显示 到 网 页 中 。 但 配合 capture 标签 使 用 后 ， 被 包 
含 的 外 部 文件 代码 并 不 会 立即 显示 ， 而 是 暂 存 变量 ， 直 到 调用 $smarty.capture.var 时 才 会 显 
未 。 如 以 下 代码 所 示 。 


<!—-{include file="mobile_head.htm1"}--> 


<!—- {capture name=foot } 一 一 > 
Eege ee ER E 
<!—-{/capture}--> 

<!-- HN SS -> 

<div data-role="content"> 


<ul data-dividertheme="þb" data-theme="c" data-role="listview" data-inset="true"> 


<!—-{foreach from=$list item=rows}--> 
<li><a href="#UserPage" ><!—-{Ş$Şrows["title"]}--></a></li> 
<!——{/foreach}——> 
le 
</div> 
<!——{S$smarty.capture.foot}-—> 


上 述 代码 首先 在 capture 标签 内 包含 foot.html 文件 ，Smarty 在 解释 时 不 会 将 代码 并 即 显 
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示 ， 而 是 保存 到 fot 变量 中 (由 name 参数 指定 ); 在 使 用 时 直接 获取 变量 名 即 可 。 利 用 
capture 标签 特性 ， 可 以 将 第 用 的 代 但 保存 到 独立 文件 ， 页 面 在 调用 时 ， 只 需要 包含 一 次 即 
可 。capture 非常 适用 于 广告 代码 、 导 航 栏 等 页 面 片段 代 但 。 


13.3.3 ”include_php (包含 PHP 文件 ) 


include 函数 只 能 包含 静态 资源 文件 ， 不 能 包含 PHP 人 代码。 虽然 大 多 数 情况 下 ， 使 用 模 
板 变量 都 能 够 完成 大 多 数 运 算 功 能 ， 但 有 些 情况 下 直接 包含 PHP 处 理 脚 本 更 加 灵活 及 简 
捷 。 例 如 局 部 页 面部 件 (Widget)、Ajax 数据 收集 、 页 面 权限 检测 等 。include_php 函数 文 持 
直接 包含 PHP 文件 代码 ， 被 包含 的 PHP 文件 会 被 服务 器 解释 ， 而 不 是 由 模板 引擎 解释 。 
include_php 标签 参数 如 表 13-1 所 示 。 


SS 13-1 include_php 参数 


参数 说 H 默认 值 
file PHP 文件 所 在 路 径 空 《 必 须 ) 
once 是 否 处 理 PHP KEUS no 

assign 使 用 变量 保存 所 包含 的 PHP 文件 代码 Ss 


如 果 设 置 了 assign 参数 ，Smarty 在 解释 时 不 会 立即 返回 结果 ， 而 是 将 运算 结果 存放 于 
assign 变量 值 中 ， 使 用 时 直接 使 用 $this.var 获取 运算 结果 ， 如 以 下 代码 所 示 。 


<l=={ include EE an assign VtestY j==> 
<l— {Ş$this.test}-—-> 


如 果 设 置 了 安全 模式 ， 被 包含 的 脚本 必须 位 于 $trusted_dir 路 径 下 。 同 时 必须 设置 file 
参数 ， 访 参数 声明 被 包含 的 PHP 文件 路 径 ， 可 以 是 $trusted_dir 的 相对 路 径 ， 也 可 以 是 绝对 
路 径 。 


13.3.4 inset (插入 函数 ) 


insert 国 数 可 以 将 现 有 的 PHP 函数 直接 插入 到 当前 模板 文件 中 ， 执 行 的 顺序 是 首先 执行 
PHP 函数 再 执行 模板 解释 。insert 函数 格式 如 表 13-2 所 示 。 


表 13-2 insert 参数 


参 H 说 H JL 值 
name 插入 的 函数 名 称 T CHW) 
assign 指定 变量 保存 插入 结果 ， 但 不 输出 T 
script 插入 函数 六 额外 执行 PHP 脚本 So 
[var..] 插入 函数 参数 ， 使 用 数组 表达 式 Sp 


insert 对 插入 的 函数 需要 使 用 关键 字 “insert ”+“name” 进 行 命 名 ， 假 设 插 入 getData 
函数 ， 代 码 如 下 。 


<!—-{insert name="getData"}--> 
AAY A E V KA A ot insert_getData， 如 以 下 代码 所 示 。 


<?php 
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function insert_getData(){ 


} 


Ge 在 插入 函数 时 文 持 为 该 函数 传 和 参数。 在 传 入 参数 时 ， 上 只 需要 直接 附加 参数 名 及 
参数 值 即 可 。 最 终 的 日 定义 函数 将 会 把 参数 转换 为 数组 键 值 对 ， 如 以 下 代码 所 示 。 

<!—-{insert name="getData" id="1" username="ceiba"}--> 

上 述 参数 的 传 入 络 采 如 以 下 代码 所 示 。 

array ("id"=>1, "username"=>"ceiba") 

所 以 ， 在 自 定义 函数 中 获取 传 入 的 参数 ， 只 需要 直接 获取 数组 键 即 可 ， 如 以 下 代码 
Dro 


function insert_getData ($array){ 
ŞuserOobj=M ("User"); 


Şuser=Ş$array [|"username"]; 


Şrows=$user0Obj->where (array ("user_name"=>$user))->find();}; 
return S$rows["user email"]; 


} 
insert 9 Sr Tu rn ARAT, ARARIRE WRIA DIS LS, 
KERT K TEE E, 13 ek ET, 


13.3.5 literal (原文 本 输出 ) 


literal 函数 用 于 原文 本 和 输出。 位 于 literal 标签 内 的 数据 ，Smarty 在 解释 时 会 按照 原 格式 
输出 。 这 些 数 据 包 括 HTML EI, PHP 代码 、Smarty 定 界 符 、JavaScript 等 脚本 。 如 以 下 代 
Droa, 

See 


<script language=javascript> 


< 
function isblank (field) { 
if (field.value == '') 
{ return false; } 
else 
{ 
document .loginform.submit (); 
EE 
} 
} 
人 
</script> 
<l Eeer =—> 


13.3.6 php GT PHP 语句 块 ) 


与 ThinkPHP FARY EE A Ce PHP 代码 不 同 ，Smarty AATE O EA e CM P Ek 
入 PHP 代码 的 ， 但 提供 了 PHP 标签 ， 用 于 实现 PHP 代码 运算 ， 功 能 类 似 于 include_php Gë 
数 。 如 以 下 代码 所 示 。 

EE 

phpinfo(); 

SE EE 
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可 以 使 用 php 标签 代 蔡 include_php 函数 ， 如 以 下 代码 所 示 。 

SE 

include ("user.php"); 

EE 

需要 注意 的 是 ， 出 于 安全 考虑 Smarty 默认 是 关闭 php 标签 支持 的 ， 如 果 需 要 开启 ， 在 
初始 化 Smarty 时 更 改 配置 即 可 。 

Şsmarty->allow_php_templates = true 


虽然 允许 艇 入 PHP RIS, (0 Smarty 官方 并 不 建议 使 用 。 


13.3.7 _ strip 保留 空格 和 回 车 符 ) 


strip 标签 用 于 格式 化 页 面 上 的 HTML 标记 ， 例 如 换行 标记 、 空 格 等 。 使 用 方法 也 比较 
简单， 如 以 下 代码 所 示 。 


EE 
<table border=0> 
E> 
<td> 
LGIA D 
<font color="red">This is a test</font> 
</A> 
</td> 
</tr> 
</table> 
Zl=={/ Strip} ==> 
上 述 代码 的 运行 结果 如 下 。 
<table border=0><tr><td><A HREF="http://my.domain.com"><font color="red">This is a 


test</font></A></td></tr></table> 


13.4 Smarty 模板 控制 语句 


Smarty 文 持 在 模板 中 使 用 判断 标签 实现 解释 流程 控制 ， 例 如 第 见 的 过 、if...else 以 及 各 
种 数据 循环 语句 等 。 下 面 将 分 别 介绍 。 


13.4.1 if, elseif (判断 语句 ) 


无 论 是 ThinkPHP 便 板 引擎 ， 还 是 Smarty 模板 引擎 ， 都 提供 了 完善 的 判断 语句 标签 。 在 
开始 介绍 判断 语句 前 ， 首 先 理解 Smarty 中 最 各 用 的 比较 运算 符 ， 如 表 13-3 rar, 


表 13-3 Smarty 常用 比较 运算 符 


Smarty 比较 运算 符 说 HH 与 PHP [匹配 
eq == 
ne, neq 不 等 于 != 
gt SE > 
lt 小 于 < 
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( 续 ) 
Smarty 比较 运算 符 说 HH 与 PHP efg 
gte、ge 大 于 或 等 于 >= 
lte, le 小 于 或 等 于 <= 
no JE ! 


Ee 的 条 件 都 是 建立 在 这 些 运 算 符 上 的 。 其 中 if 判断 语句 是 最 常用 的 流程 控制 语句 
， 在 Smarty 中 ， 所 有 流程 控制 语句 均 使 用 标签 表示 ， 如 以 下 代码 所 示 。 


<!—-{if $class eq "news" 1} 一 一 > 
<!—-{foreach from=$list item=rows}--> 
<li><a href="#UserPage" ><!-—-{Şrows["title"]}--></a></li> 
<!——{/foreach}--> 


<!—-{/if}-—-> 
对 应 的 后 台 代 人 码 如 下 。 
StaiLls->smarty->assicm ee 


上 上 述 代码 表示 根据 后 人 台 的 GET RK, HENTEZ “news”. WRAEK, MAA 
行 数据 循环 操作 。 

与 PHP 一 样 ， 存 在 if 语句 ， 当 然 也 会 存在 elseif K else 语句 。 在 Smarty 中 同样 有 对 应 
的 标签 ， 如 以 下 代码 所 示 。 


<!—-{if $class eq "news" 1} 一 一 > 
<!—-{foreach from=$list item=rows}--> 
<li><a href=" p UserPage" >< l=- 1 Srows | ticle” ] ==></a></1i> 
<!——{/foreach}——> 


<!—{elseif Ge eq "sports"}--> 
体育 频道 尚未 开 

<!—{else}--> 
参数 请 求 错误 

<!-—-{/if}--> 


13.4.2 foreach 〈 循 环 数据 ) 


在 前 面 的 内 容 中 ，foreach 标签 已 经 多 次 出 现 ， 相 信 谈 者 已 经 能 够 掌握 其 使 用 方法 。 
foreach 是 Smarty 模板 引擎 中 最 简单 的 循环 语句 标签 ， 该 标签 属性 如 表 13-4 所 示 。 


表 13-4 foreach 属性 


属 性 说 W Æ T 2 

form 循环 数组 数据 源 是 

item 循环 临时 存放 变量 是 

key 当前 数组 key EN 

name 循环 源 名 称 ， 用 于 访问 循环 源 T 

其 中 属性 key 是 一 个 隐匿 的 数组 和 变量， 在 循环 时 直接 获取 即 可 ， 如 以 下 代码 所 示 。 
<!—-{foreach from=$list item=rows key=Kkey } 一 一 > 
<li>Key: <!—{ $key} --> ËR: <!—-{$rows[|"title"]}--></li> 
<!—-{/foreach}--> 


需要 注意 的 是 ，foreach 是 以 属性 name 为 循环 依据 的 ， 如 果 访 属性 值 为 空 ，foreach 将 退 
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13.4.3 section 〈 遍 历数 组 ) 


与 foreach 一 样 ，section 标签 也 是 用 于 循环 数组 数据 的 ， 但 foreach 只 能 循环 简单 的 一 维 
数据 ， 例 如 索引 数组 、 关 联 数组 等 。 而 section 不 仅 能 够 循环 单 维 、 多 维 数 组 ， 还 支持 无 限 
PAREMA. section 标签 属性 如 表 13-5 所 示 。 


SS 13-5 section 属性 


属 性 说 H Æ D 2 
name 循环 名 称 是 
loop 循环 数组 数据 源 是 
start Iëik Ire, AA 0 S 
step 循环 步 长 ， 即 本 次 循环 与 上 次 循环 中 间 跳 过 的 记录 数 "e 
max 最 大 循环 次 数 ， 默 认 情 况 下 根据 数组 记录 决定 次 数 E 
Show 是 否 显示 循环 E 


事实 上 section 被 编译 后 ， 就 是 PHP 中 的 for 语句 ， 如 以 下 代码 所 示 。 

<!—-{section name=i Loop=$11s 七 } 一 一 > 
<li><a href=-"}UserPage" ><" ={sListlil titlel ></a></li> 

<!—-{/section}--> 

上 述 代 人 码 编译 后 的 PHP 代码 如 以 下 代码 所 示 。 

<?php 

Şlist= array ( 

0=>array ( 

"id"=>1, 

"title"=> "标题 1"， 
), 
1=> array ( 

tidi], 

"title"=> "标题 2"， 
) ， 

) 7 

ENEE EE ($SLisSt) SLF) 1 

el a Te 

} 

相信 阅读 本 书 的 读者 都 能 够 看 懂 上 述 代码 ， 所 以 这 里 不 再 说 明 。 前 面 介绍 的 section 标 
签 ， 被 编译 后 的 代码 形式 就 如 上 述 代 码 所 示 。 当 然 在 真实 的 生产 环境 中 ， 数 据 源 通 常 来 自 于 
后 台数 据 库 〈 即 后 台 PHP 变量 )， 这 里 为 了 便于 讲解 ， 简 化 了 代码 。 

从 这 就 可 以 看 出 ， 属 性 name 相当 于 for 循环 的 条 件 变量 〈 即 $); 属性 loop 相当 于 数组 
存放 变量 〈 即 $list);， 属 性 start 相当 于 起 始 偏 移 量 〈 即 $i=0); 属性 max 相当 于 循环 结束 数量 
《 即 $i<count($lisb); step 相当 于 循环 步 进 ( 即 $i++)。 通 过 这 样 的 比较 ， 相 信 读 者 就 能 轻松 掌 
握 section 标识 的 使 用 了 。 

section 标签 基于 for 语句 实现 循环 ， 而 foreach 标签 基于 foreach 语句 实现 循环 ， 所 以 
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foreach 在 性 能 上 具备 优势 。 在 实际 应 用 开发 中 可 根据 需要 进行 选择 。 


13.5 ”变量 调节 器 


变量 调节 器 也 称 变量 修饰 函数 ， 它 是 Smarty 内 置 的 功能 类 库 ， 方 便 开 发 人 员 直 接 调 
用 。 在 使 用 方式 上 与 ThinkPHP 中 的 标签 外 部 尔 数 闫 似 。 下 面 将 对 Smarty RV OI em 
用 变量 调节 器 进 行 介绍 。 


13.5.1 capitalize 〈 首 字母 大 写 ) 
capitalize 函数 可 以 实现 英文 首 学 母 大 写 ， 但 对 中 文字 符 没 有 意义 。 如 以 下 代码 所 示 。 


Şthis->smarty->assign('topic', 'Police begin campaign to rundown jaywalkers.!'); 


Şthis->smarty->display ('index.html'); 


index.html 模板 代码 如 下 。 


<!—--{$topic|capitalize}--> 
最 后 输出 的 结 朱 如 下 。 


Police Begin Campaign To Rundown Jaywalkers. 


13.5.2 count characters (统计 字符 ) 
count_characters 国 数 可 以 统计 变量 中 的 字符 数量 ， 文 持 中 文 统计 。 


Şthis->smarty->assign('topic', ‘Hello PHP5'); 
Sthis->smarty->assign('pageTit1le' vv 欢迎 光临 ") ; 
Şthis->smarty->display ('index.html'); 


index.html 模板 代码 如 下 。 


Number1:<!--{$topic|count_characters }--> 


Number2:<!--{$pageTitle|count_characters}--> 
最 后 输出 的 结果 如 下 。 

Numberl: 9 

Number2: 4 


13.5.3 count_paragraphs“〈 统 计 段 落 ) 


count_paragraphs 图 数 是 一 个 根据 回 车 换行 符 得 到 文件 段 沙 数量 的 变量 调 季 事 ， 文 持 
中 文 。 
sthis->smarty->assign('topic',，'Smarty 并 不 是 一 个 新 的 PHP 技术 ， 而 是 在 PHP4.x 以 前 就 已 经 是 非 
常 流行 的 界面 代码 与 PHP 代码 相 分 离 的 技术 。 
由 于 PHP 的 特性 ， 早 期 的 PHP 编程 要 实现 MVC 编程 ， 是 非常 困难 的 事 ， 但 Smarty 出 现 之 后 ...')， 
Şthis->smarty->display ('index.html'); 


index.html 模板 代码 如 下 。 


Paragraph:<!--{$topic|count_paragraphs}--> 
最 后 输出 结果 如 下 。 


Paragraph: 2 
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13.5.4 count sentences 〈 统 计 句 数 ) 


count_sentences 卫 数 可 以 统计 变量 中 的 句 数 ， 所 谓 的 句 数 是 指 英 文 语法 中 的 句 数 ， 即 英 


"VI GE MEF. 


"); 


格 ， 


Şthis->smarty->assign('topic', 'beauty-soft.net. www.beauty-soft.net.!'); 


$this->smarty->assign ('pageTitle',".net 开发 相 比 PHP 开发 更 加 复杂 . (EREEREER . FA 


Şthis->smarty->display ('index.html'); 


index.html 模板 代码 如 下 。 


<!--{$topic|count_sentences}--> 
<DE 
<!—-{$pageTitle|count_sentences}--> 
最 后 输出 结果 如 下 。 

2 

] 


可 以 看 到 ， 尽 管 pageTitle 设置 了 3 个 英语 氮 号 〈.)， 但 由 于 上 只 有 其 中 一 个 英文 连接 罕 
所 以 统计 结束 为 2。 


13.5.5 count words 〈 统 计 单 词 ) 


count_words 可 以 统计 变量 中 的 殉 文 单词 数量 ， 不 文 持 中 文 拼音 或 汉字 。 
Şthis->smarty->assign('topic', 'beauty-soft.net. www.beauty-soft.net.!'); 
$this->smarty->assign ('pageTitle', "欢迎 光临 本 站 ,Welcome to php"); 
Şthis->smarty->display ('index.html'); 


index.html 模板 代码 如 下 。 


<!--{$topic|count_words}--> 
SE 
<!--{$pageTitle|count_words}--> 
最 后 输出 结 采 如 下 。 

5 

4 


13.5.6 date format (格式 化 日 期 》 


date_format 函数 可 以 格式 化 变量 中 的 日 期 及 时 间 。 访 图 数 共 文 持 2 个 参数 : 第 1 个 参数 


表示 输出 日 期 的 格式 ; 第 2 个 参数 为 可 选 参数 ， 表 示 输 入 时 参数 的 日 期 及 时 间 格 式 。 在 
Smarty 引擎 中 ， 变 量 调节 器 函数 参数 乙 间 使 用 “;” 分 隔 符 。 其 中 参数 1 表示 输出 格式 ， 
Smarty 使 用 特定 的 字符 表示 格式 ， 和 常用 的 格式 如 表 13-6 Dir or, 


表 13-6 date format 时 间 格 式 


字 符 串 输出 格式 效果 样 例 
%a 当前 区 域 星 期 几 的 简写 Thu 
%A 当前 区 域 星期 几 的 全 称 Thursday 
%b 当前 区 域 月 份 的 简写 Nov 
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CZE) 
字 符 串 输出 格式 效果 样 例 

%B 当前 区 域 月 份 的 全 称 November 

Ge 当前 区 域 首 选 的 日 期 时 间 表 达 11/29/12 11:11:56 

%d 日 号 的 十 进 制 数 表达 (范围 从 1 一 31) 30 

Gm 月 份 的 十 进 制 数 表达 范围 从 1 一 12) 10 

%H 24 ERRI F RER ya EA 00~23) 23 

ell 12 小 时 制 的 十 进 制 小 时 数 《〈 范 围 从 00 一 12) 11 

%S 十 进 制 秒 数 44 

ou 星期 儿 的 十 进 制 数 表达 3 


事实 上 date_format 编 详 后 就 是 PHP 中 的 strftime 子 数 。 接 收 的 时 间 格 式 可 以 是 PHP 中 


有 效 的 时 间 ， 例 如 time、date 等 。 


Şthis->smarty->assign('pdate',time()); 
Şthis->smarty->display ('index.html'); 


index.html 模板 代码 如 下 。 


现在 时 间 : <!--{$pdate|date_format:"%Y Æ sm 月 $d 日 $8H 点 $I 分 $S 秒 "}--> 
最 后 输出 结果 如 下 。 
现在 时 间 : 2012 年 10 月 28 日 20 点 11 分 14 秒 


13.5.7 escape (字符 转 码 ) 


酒 。 


escape 能 够 对 得 出 变量 进行 转 码 ， 与 PHP 中 的 urlencode 函数 类 似 ， 但 使 用 方式 比较 有 灵 
escape 函数 通过 改变 参数 的 值 改 变 转 码 方式 ， 参 数值 如 表 13-7 Dirr, 


表 13-7 escape 参数 


ZS 数 说 D 示 例 
html 使 用 htmlspecialchars 过 滤 HTML {$articleTitlelescape:"html"} 
htmlall 使 用 mb_convert_encoding 过 滤 HTML {$articleTitlelescape:"htmlall"} 
url 使 用 urlencode Gë SEA {$articleTitlelescape: "url"} 
quotes 使 用 preg_quote 正则 过 滤 引 用 代码 {$articleTitlelescape:"quotes"} 
hex 加 密 电子 邮件 地 址 {$emaillescape:"hex"} 


13.5.8 replace (字符 替换 ) 


数 : 


replace 国 数 可 以 方便 地 对 输出 变量 进行 学 符 玲 换 ， 文 持 中 文学 符 。 该 玫 数 共 文 持 两 个 参 
参数 1 表示 需要 符 换 的 原 字 人 符 ;参数 2 表示 替换 后 的 字符 。 
Şthis->smarty->assign('topic', 'beauty-soft.net. www.beauty-soft.net.!'); 
$this->smarty->assign ('pageTitle', "欢迎 光临 本 站 ,Welcome to php"); 
Şthis->smarty->display ('index.html'); 


index.html 模板 代码 如 下 。 


<!--{$pageTitle|replace: "欢迎 ":"Welcome"}--> 
br /> 
er ee aee e re 
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最 后 输出 结 朵 如 下 。 
Welcome 光临 本 站 ，Welcome to php 


beauty-soft.net www.beauty-soft.net 


13.5.9 regex_replace 〈 正 则 替换 ) 


regex_replace 与 replace 函数 功能 类 似 ， 都 是 用 于 符 换 变量 输出 的 字符 ， 不 同 的 是 
regx_replace 图 数 使 用 正则 表达 式 处 理 航 符 换 的 字符 。 


ŝthis -smarty -assign('topic', "haii: [hide]beauty soft.net|/hide]"); 
Şthis->smarty->display ('index.html'); 


index.html 模板 代码 如 下 。 


<!--{$topic|regex_replace:"/\[hide\] (.*?)\[\/hide\]/i":"* 内 容 回 复 可 见 *"}---> 
最 后 输出 结果 如 下 。 
下 载 地 址 : * 内 容 回复 可 见 # 


13.5.10 truncate (字符 截取 ) 


truncate 闲 数 可 以 对 输出 变量 进行 学 答 截 取 ， 功 能 与 str_replace PHP RAKI. truncate 
ELE 3 个 参数 。 其 中 参数 1 为 必 选 项 ， 表 示 和 截取 的 字符 数量 《中文 以 双 字 节 进 行 计 算 ); 
参数 2 为 可 选择 项 ， 表 示 截 取 后 退 加 的 字符 ;参数 3 为 可 选项 ， 表 示 是 否 截 取 到 单词 边界 ， 
默认 为 false。 


$this->smarty->assign ('pageTitle', "欢迎 光临 本 站 ,Welcome to php"); 
Şthis->smarty->display ('index.html'); 


index.html 模板 代码 如 下 。 


le 

<br/> 

Eee Tl lueae on > 

< / > 
人 


最 后 输出 结果 如 下 。 
t1: 欢 迎 光 临 本 .. 

人 :欢迎 光临 本 站 ，W 
t3: 欢 迎 光 临 本 .… 


13.6 ”视图 助手 


Smarty 内 置 了 一 系列 水 数 用 于 生成 HTML 表单 组 件 ， 这 些 函 数 称 为 视图 助手 。 视 图 助 
手 本 质 上 是 方便 开发 人 员 以 更 简洁 的 代码 代 蔡 HTML 代码 ， 从 而 提高 开发 效率 。 币 用 的 视 
图 助手 包含 html_options、html_radios、html_select_date 等 。 下 面 分 别 介 绍 。 


13.6.1 bim Image 〈 生 成 图 像 ) 
html_image 标签 用 于 生成 图 像 探 件 。 该 标签 共 文 持 7 个 属性 ， 如 表 13-8 所 示 。 
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SS 13-8 ml image 属性 


属 性 说 H 默 认 值 
file 图 像 的 地 址 (本 地 或 远程 均 可 ) 是 (必须 ) 

border 图 像 边 框 大 小 

height 图 像 高 度 Da 

width 图 像 宽度 目 动 

basedir 文件 相对 路 径 当前 网 站 根 目 录 
alt WERE E D T 

href 图 像 链 接地 址 T 


html_image 标签 中 的 各 项 属性 与 html imge 标签 元 素 一 致 ， 如 以 下 代码 所 示 。 


<!—-{html_image file="http://www.beauty-soft.net/soft-css/t17.png" wicien=” 100% 
heLches= "30W ==> 


最 终生 成 后 的 HTML 如 以 下 代码 所 示 。 

<img width="100" height="30" alt="" src="http://www.beauty-soft.net/soft-css/t17. 
png"> 

需要 说 明 的 是 ， 虽 然 html_image 标签 可 以 生成 HTML 图 像 控件 ， 但 其 本 质 上 最 终 还 是 
由 PHP 转换 的 ， 这 就 意味 着 过 多 地 使 用 html_image 标签 ， 将 会 导致 解释 引擎 的 性 能 损失 。 
所 以 在 实际 应 用 开发 中 ， 应 尽量 避免 使 用 html_image 标签 。 


13.6.2 html_options (生成 表单 选择 组 件 ) 


html_options 标签 能 够 根据 后 台 的 数据 源 生成 HTML 下 拉 选 择 组 件 ， 很 好 地 代 符 foreach 
循环 。html_options 标签 共 支 持 5 个 属性 ， 如 表 13-9 所 示 。 


SS 13-9 html_options 属性 


属 性 说 Di S 认 值 
value 下 拉 列 表 选 择 项 对 应 的 值 ， 需 要 确保 唯一 性 A CAE) 

output 下 拉 列 表 选 择 项 文本 T OUR) 
selected 默认 组 件 选中 的 值 AORE) 

options 生成 组 件 的 数据 源 〈 关 联 数组 ) 宇 OË) 

name 组 件 名 称 空 〈 必 选 ) 


需要 注意 的 是 ， 如 果 value 属性 值 为 宇 ，html_options 将 使 用 数组 循环 key 作为 属性 值 。 
下 和 面 结合 示例 代码 ， 演 示 html_options 标签 的 使 用 。 
public function Inaex() { 
sarticle=M ("Article"); 


Şrows=$article->where (array ("add user"=>"ceiba"))—>select () ; 


Soptions_ rows=array ();}; 
foreach ($rows as $key => Svalue) { 
array_push ($options_rows, $value["title"]); 


} 


$options_ rows_ id=array (); 


foreach ($rows as $key => Svalue) { 
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array_push ($options_rows_id, Ş$value["id"]); 
} 
Şthis->smarty->assign('options_rows_id', $options_rows_id); 
Şthis->smarty->assign('cust_options', $options_rows);// output 
“ehis -snart -assigni customer I soptilons rows Idilo scele ted 


// values 


Şthis->smarty->display ('index.html'); 
} 


index.html 模板 代码 如 下 。 


<select name=customer id> 


<!—-{html_options values=$options_ rows_id output=$cust options selected= Scustomer 
_ 6h ==S 

</select> 

,二 二 六 e 

运行 效 末 如 图 13-5 Dirr, 


Firefox ~ 
C} http;//tp.loca...st/mobile. php#| +| 
e+ UU Gpe] corn Bi 


库 克 就 苹果 地 图 向 陪 向 用 户 道 浴 : 8 SH e 
Kowa CO: BD Ste 


KAEEREief FAKE OC EK d E 
夏普 称 合 制 利 足 和 能 iPhones5 所 需 的 显示 屏 

蔷 果 流 曙 体 电 音 服 务 遇 障 在 : 索尼/ATY 对 昼 格 不 医 
NARE Facebook t AA Ateni 
FEENS SPHE KANALAS 


图 13-5 html options 生成 效果 


最 终生 成 的 HTML 如 以 下 代码 所 示 。 


<select name=customer_id> 


<option value="1" selected="selected"> JE m MLY RH K| ik K m HAER: 未 提供 一 流体 验 


</option> 
<option value="2 "> RERA 6.44 亿美 元 成 为 奥 林 巴 斯 第 一 大 股东 </opticon> 
<option value="3"> 夏 普 称 会 制造 足够 iPhone5 所 需 的 显示 屏 </option> 
<option value="4"> 苹 果 流 媒体 电台 服务 过 障碍 :索尼 /ATV 对 价格 不 满 </option> 
<option value="5"> 机 构 调 查 称 Facebook 中 国 用 户 数 超 6000 万 </option> 
<option value="6"> 李 并 宏 称 仍 未 找到 适合 移动 互联 网 的 商业 模式 </option> 


</select> 


13.6.3 html radios 〈 生 成 表单 单 选 组 件 ) 


html_radios 标签 与 html_options 标签 一 样 ， 都 是 根据 后 台数 据 源 生成 相应 的 表单 元 系 。 
html_radios 用 于 生成 表单 单 选 按钮 ， 该 标签 共 文 持 6 个 属性 ， 如 表 13-10 所 示 。 


SS 13-10 html radios 属性 


属 性 说 H S A 值 
name 单 选 按钮 名 称 TEA) 
values 单 选 按钮 值 〈 需 要 唯一 性 ) "7 LD" 
output 单 选 按钮 的 显示 文本 E LE 
checked 选 定 的 单 选 按钮 值 空 〈 可 选 ) 
options 后 台数 据 源 (关联 数组 ) Z OYE) 
separator 单 选 按钮 分 隔 符 a CD 
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接 下 来 继续 以 前 面 创建 的 数据 源 为 基础 ， 介 绍 html _ radios 标签 的 实际 应 用 。 这 里 内需 
要 修改 index.html 模板 文件 代码 即 可 。 


<!—-{hħhtml_radios name="id" options=$cust_ options checked=$customer_id separator= 
"kr />"}——> 


其 中 separator 属性 支持 合法 的 HTML 标签 ， 运 行 效 果 如 图 13-6 所 示 。 


Firefox = 
[i http://tp.localhost/mobile.php 


e > EE BEE 


© 库 克 就 苹果 地 图 缺陷 向 用 户 道歉 : 未 提供 一 流体 验 
加 @ 素 尼 投 资 6.44 亿 美元 成 为 奥 林 巴 斯 第 一 大 股东 


辐 豆 普 称 会 制造 足够 iPhone5 所 需 的 显示 屏 

轩 苹果 流 媒 体 电 台 服 务 遇 障碍 : 索尼 /ATV 对 价格 不 满 
国 机 构 调查 称 Facebook 中 国 用 户 数 超 6000 万 

看 地 彦 宏 称 仍 未 找到 适合 移动 互联 网 的 商业 模式 


图 13-6 html radios 生成 效果 
最 终生 成 的 HTML 如 以 下 代码 所 示 。 


<label><input type="radio" name="id" value="0" /> 

库 克 就 华 果 地 图 缺陷 向 用 户 道歉 : 未 提供 一 流体 验 </label><br /> 

<label><input type="radio" name="id" value="1" checked="checked" /> 
索尼 投资 6 A4 亿美 元 成 为 奥 林 巴 斯 第 一 大 股东 </1label><br /> 

<label><input type="radio" name="id" value="2" /> 

夏普 称 会 制造 足够 iPhone5 所 需 的 显示 屏 </Labe1l><br /> 

<label><input type="radio" name="id" value="3" /> 

苹果 流 媒 体 电台 服务 过 障碍 : 索尼 /RATV 对 价格 不 满 </label><br /> 

<label><input type="radio" name="id" value="4" /> 

机 构 调查 称 Facebook 中 国 用 户 数 超 6000 万 </label><br /> 


<label><input type="radio" name="id" value="5" /> 


李 谍 宏 称 仍 未 找到 适合 移动 互联 网 的 商业 模式 </Labe1L><pbr /> 


13.6.4 html checkboxes (生成 表单 复 选 组 件 ) 


html_checkboxes 标签 用 于 方便 生成 表单 复 选 按钮 ， 使 用 方式 上 与 前 面 介 绍 的 
html_radios 和 html_options 相 类 似 。html_checkboxes 标签 所 支持 的 属性 如 表 13-11 所 示 。 


表 13-11 html checkboxes 属性 


属 性 说 明 默 认 值 

name 复 选 按钮 名 称 a C) 
values 复 选 按钮 值 〈《 需 要 唯一 性 ) a E) 
optput 复 选 按钮 显示 文本 E OA) 
selected 选 定 的 复 选 按钮 值 空 〈 可 选 ) 
options 后 台数 据 源 ( 关 联 数 组 ) Se lee 
separator 复 选 按钮 分 隔 符 T (可 选 ) 
labels 是 否 为 每 个 复 选 按钮 添加 labels 标签 true 
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继续 以 前 面 创建 的 数据 源 为 基础 ， 详 细 介 绍 html_checkboxes 标签 的 实际 应 用 。 这 里 只 
需要 修改 index.html 模板 文件 代码 即 可 。 


<!—-{html_checkboxes name="checkboxes id" options =$cust options checked=$customer _ id 
separator="<br />"}-——> 


运行 效果 如 图 13-7 所 示 。 


区 http://tp.localhost/mobile.php| + | 


é> Ü G [@ tpv e|- ccr] t E 


E EAEE RAA AER : 未 提供 一 流体 验 
国 素 尼 授 资 6.44 亿 美元 成 为 奥 林 巴 斯 第 一 大 股东 


套 夏 普 称 会 得 | 造 足 够 iPhone5 所 零 的 显示 屏 

加 苹果 流 媒体 电台 服务 负 障 碍 : 素 尼 /ATV 对 价格 不 满 
回 机 构 调 查 称 Facebook 中 国 用 户 数 超 6000 万 

加 李彦宏 称 仍 未 找到 适合 移动 互联 网 的 商业 模式 


图 13-7 html checkboxes 生成 效果 


最 终生 成 的 HIMEL 如 以 下 代码 所 示 。 


<label><input type="checkbox" name="id[]" value="0" /> 
Fr rat WR hënt IO "rëm tel ebr /> 
<label><input type="checkbox" name="id[]" value="1" checked="checked" /> 
索尼 投资 6.44 亿美 元 成 为 奥 林 巴 斯 第 一 大 股东 </1label><br /> 
<label><input type="checkbox" name="id[]" value="2" /> 
夏普 称 会 制造 足够 iPhone5 所 和 需 的 显示 屏 </label><br /> 
<label><input type="checkbox" name="id[]" value="3" /> 
苹果 流 媒 体 电台 服务 过 障 碍 : 索尼 /ATV 对 价格 不 满 </1abel><br /> 
<label><input type="checkbox" name="id[]" value="4" /> 
机 构 调查 称 Facebook 中 国 用 户 数 超 6000 万 </label><br /> 
<label><input type="checkbox" name="id[]" value="5" /> 


李彦宏 称 仍 未 找到 适合 移动 互联 网 的 商业 模式 </Labe1L><pbr /> 


13.6.5 html select date 〈 生 成 表单 日 期 选择 组 件 ) 


html_select_date 标签 用 于 方便 生成 日 期 选择 组 件 。 这 里 所 说 的 日 期 选择 实际 上 就 是 
select 下 拉 选 择 组 件 ， 不 同 之 处 在 于 html_select_date 标签 能 够 智能 填充 数据 ， 吕 免 设计 人 员 
手动 输入 ， 以 提高 开发 效率 。html_select_date 标签 文 持 的 属性 非常 多 ， 这 里 只 列 出 重要 及 名 
用 的 属性 ， 如 表 13-12 所 示 。 


表 13-12 html select date 属性 


属 性 说 。 明 默 认 值 
prefix 变量 名 称 前 级 空 〈 必 选 ) 
time UNIX 时 间 戳 空 〈 可 选 ) 
start_year 下 拉 列 表 开 始 年 份 全 
end_year 下 拉 列 表 结 束 年 份 SSC 
display_days 是 合 显示 天 true 
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CÈ) 
Sa 性 说 明 默 认 值 
display_months 是 全 显示 月 true 
display_years 是 否 显示 年 true 
month_format 基于 strftime 函数 的 月 份 格 式 SS 
day_format 基于 strftime 函数 的 日 号 格式 E 
daa value Format 月 号 显示 格式 T 
year_as_text 是 耕 以 文本 形式 显示 年 份 false 
reverse_years 逆序 显示 年 份 false 
接 下 来 将 通过 代码 演示 html_select_date 标签 的 使 用 ， 如 以 下 代码 所 示 。 
<la select are Drerix—"ScarDare, cimae=SEime start Year=— 3 en yeanr=W iY 


display_days=false}--> 


上 上述 代码 的 显示 效果 如 图 13-8 PZR. 


Firefox = 


[i http://tp.localhost/mobile.php 


e> UU [aprel ccr A 加 


September 
et ober 

November 
December 


图 13-8 html select date 生成 效果 


最 终生 成 的 HTML 如 以 下 代码 所 示 。 


<select name="StartDateMonth"> 


<option value="01">January</option> 
<option value="02">February</option> 
<option value="03">March</option> 
<option value="04">April</option> 
<option value="05">May</option> 
<option value="06">June</option> 
<option value="07">July</option> 
<option value="08">August</option> 
<option value="09">September</option> 
<option value="10">0ctober</option> 
<option value="11">November</option> 
<option value="12">December</option> 


</select> 


<select name="StartDateYear"> 


<option 
<option 


value="2007">2007</option> 
value="2008">2008</option> 
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<option value="2009">2009</option> 

<option value="2010">2010</option> 

<option value="2011">2011</option> 

<option value="2012">2012</option> 

<option value="2013">2013</option> 
</select> 


13.6.6 html select time 〈 生 成 表单 时 间 选 择 组 件 ) 


html_select_time 标签 与 html_select_date 标签 一 样 都 是 用 于 生成 时 间 组 件 的 ， 不 同 的 是 
html_select_time 以 小 时 为 单位 ， 包 括 12 小 时 模式 及 24 小 时 模式 。html_select_time 支持 的 属 
性 如 表 13-13 所 示 。 


SS 13-13 html select time 属性 


属 性 说 明 LL 值 
prefix 组 件 的 变量 名 前 级 Time_ 
time UNIX DInlëk 空 〈 可 选 ) 
display_hours 是 人 否 显示 小 时 true 
display_minutes 是 否 显示 分 钟 true 
display_seconds 是 否 显示 秘 true 
display_meridian re LETTI) true 
use 24 hours 是 否 开 局 24 小 时 制 true 
minute_interval 分 钟 下 拉 列 表 的 间隔 1 
second_interval 秒 钟 下 拉 列 表 的 间隔 1 
field_array 输出 值 到 该 值 指定 的 数组 SEO 


接 下 来 将 通过 代码 演示 bm! select time 标签 的 使 用 ， 如 以 下 代码 所 示 。 
<!—-{html_select_time use 24 hours=true prefix="my_"}--> 


运行 效果 如 图 13-9 所 示 。 


图 13-9 html select_time 生成 效果 
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13.6.7 html table 〈 生 成 表格 ) 
html_table 能 够 根据 后 侣 的 数据 源 动 态 生成 表格 视图 。 所 文 持 的 属性 如 表 13-14 所 示 。 


SS 13-14 html table 属性 


属 性 说 明 默 认 值 
loop 后 台数 组 数据 源 E A) 
cols 表格 列 数量 3 
table_attr 表格 线条 边框 宽度 1 
tr_attr 行 标签 属性 SOEN 
td_attr 列 标签 属性 人 
trailpad 最 后 一 行 附加 的 数据 空 〈 可 选 ) 
hdir 表格 行 对 齐 方式 ， 可 选 的 值 有 left 和 right left 
vdir 表格 列 对 齐 方式 ， 可 选 的 值 有 up 和 down down 


接 下 来 将 通过 代码 演示 html_table 标签 的 使 用 ， 如 以 下 代码 所 示 。 


public function index (){ 
Şthis->smarty->template_dir=APP_PATH.'/Tpl/index'; 
Ş$article=M ("Article"); 
Şlist=array (); 
foreach ($rows as $key => $value) { 
array Dúsh (gliste, $value [Yie ]) } 
array püsh ($list, $value "riecleY]) z 
} 
Şthis->smarty->assign("rows",Ş$list); 
Şthis->smarty->display ('index.html'); 
} 


index.html 模板 代码 如 下 。 


<!—-{html_table loop=$rows cols=2 table attr='border="0" cellspacing="1" '}-—--> 
上 上 述 代码 的 运行 效 末 如 网 13-10 Dr ar, 


Firefox = 
e http://tp.localhost/mobile.php 
é> RH DG'em/eceug- emp 


1 库 克 就 苹果 地 图 缺陷 向 用 户 道 烙 : 未 提供 一 流体 异 

2 索尼 投资 6.44 亿 美元 成 为 奥 林 巴 斯 第 一 大 股东 

3 夏普 称 会 制造 足 能 iPhone5 所 需 的 显示 屏 

4 苹果 流 媒体 电台 服务 遇 障 碍 : 素 尼 /ATV 对 价格 咎 满 
5 机 构 调查 称 Facebook 中 国 用 户 数 超 6000 万 

6 李彦宏 称 仍 未 找到 适合 移动 互联 网 的 商业 模式 


图 13-10 html table 生成 效果 
最 终生 成 的 HTML 如 以 下 代码 所 示 。 


<table border="0" cellspacing="1" > 
<tbody> 

LTES 

<td>1</td> 
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<td> 库 克 就 苹果 地 图 缺陷 向 用 户 道歉 : 未 提供 一 流体 验 </tad> 
tr 

LTES 

<td>2</td> 

<td> 索 尼 投 资 6.44 亿美 元 成 为 奥 林 巴 斯 第 一 大 股东 </td> 
L/ TES 

LTES 

<td>3</td> 

<td> 夏 普 称 会 制造 足够 iPhone5 所 需 的 显示 屏 </td> 
SE 

LTES 

<td>4</td> 

<td> ERMAR S IER: 索尼 /ATV 对 价格 不 满 </tq> 
SHEET 

SE 

< 七 q>5< /七 Q> 

<td> 机 构 调 查 称 Facebook 中 国 用 户 数 超 6000 万 </td> 
</tr> 

KETS 

<td>6</td> 

<t OFE EC Er UNK REESE HRA E dd tad> 
L/T ES 

</tbody> 

</table> 


13.7 Smarty 对象 方 法 


前 面 对 Smarty 模板 引 敬 常见 的 视图 标签 进行 了 详细 介绍 ， 这 些 标签 或 水 数 对 设计 人 员 
非常 重要 ， 对 提高 网 站 的 开发 效率 非常 有 帮助 。 接 下 来 将 介绍 Smarty 对 象 对 外 提供 的 常用 
方法 ， 通 过 调用 或 改变 Smarty 对 象 方法 值 ， 可 以 提高 Smarty 的 灵活 性 ， 例 如 注册 目 定 义 函 


13.7.1 display (显示 模板 ) 


display 方法 在 前 面 的 内 容 中 己 经 多 次 涉及 ， 相 信人 读者 已 经 非常 熟悉 。display 方法 用 于 
显示 视图 模板 ， 共 支持 4 个 参数 ， 形 式 如 下 。 

public function display($template = null, Scache wd = null, $compile_id = null, $parent = null) 

其 中 参数 template 表示 模板 文件 路 径 ， 文 持 相 对 路 径 及 绝对 路 径 ， 模 板 文件 后 缀 名 为 
“tp” 参数 cache id 表示 绥 存 id; 参数 compile_id 表示 编译 id， 通 党 情况 下 一 僚 模 板 对 应 一 
个 编译 写 〈 由 系统 目 动 分 配 )， 但 如 来 需 要 一 套 模板 应 用 在 多 种 环境 例如 中 英文 切换 )， 这 
时 compile_id 就 派 上 用 场 了 ; 参数 parent 表示 取 模 板 范围 。 下 面 将 通过 示例 演示 display 的 
使 用 。 

// 绝对 路 径 

Şsmarty->display ("/usr/local/include/templates/header.tpl"); 

// 绝对 路 径 〈 另 外 一 种 方式 ) 

$smarty->display ("file:/usr/local/include/templates/header.tpl"); 


// WINDOWS 平台 下 的 绝对 路 径 〈 必 须 使 用 “file:” 前 绥 ) 
Şsmarty->display ("file:C:/www/pub/templates/header.tpl"); 
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13.7.2 fetch (获取 输出 内 容 ) 


fetch 方法 与 display 方法 类 似 ， 用 于 输出 模板 。 但 fetch 不 会 直接 输出 ， 而 是 将 结果 保存 
到 临时 变量 。fetch 方法 形式 如 下 。 


public function fetch($template = null, Scache id = null, $compile_id = null, $parent = null, 


$display = false, $merge_tpl_vars = true, $no_output_filter = false) 

其 中 参数 template 表示 模板 文件 ， 参 数 cache_id 表示 模板 缓存 id; 参数 compile id 表示 
编译 id; 参数 parent 表示 取 模 板 范围 ;参数 display 表示 是 否 直接 显示 。 看 到 display 参数 ， 
相信 读者 已 经 能 够 明白 fetch 与 display 之 间 的 关系 了 。 

事实 上 display Wik fetch 的 功能 重 写 ， 通 过 改变 参数 display 实现 模板 直接 显示 。fetch 
虽然 较 少 使 用 ， 但 在 一 些 权限 认证 的 网 页 中 还 是 会 用 到 的 ， 例 如 以 下 代码 所 示 。 


class IndexAction extends PublicAction { 


public function index (){ 
Şthis->smarty->template_dir=APP_PATH.'/Tpl/index'; 
Ştmp=$this->smarty->fetch('index.html'); 


if (session ("username")){ 

ea 

$stmp=str_replace("[/name/]", session ("username"), $tmp); 
}else{ 


ane oae a p EE 
} 
echo $tmp; 


} 

fech 将 结果 保存 到 变量 中 ， 开 发 人 员 可 以 直接 和 输出， 也 可 以 将 模板 数据 保存 到 数据 
库 、 集 群 绥 存 系统 中 。 在 执行 输出 或 保存 本 ， 开 发 人 员 可 以 方便 地 对 数据 进行 修改 、 增 加 、 
删除 等 。 从 这 一 点 来 分 机，fetch 非常 适合 输出 xml、wap 等 模板 。 


13.7.3 configLoad 〈 加 载 配置 信息 ) 


configLoad 方法 用 于 加 载 配置 文件 信息 。 其 中 参数 1 表示 配置 文件 地 址 (相对 于 
$smarty->config dr 指定 值 );， 参数 2 表示 加 载 的 配置 和 节点。 假设 配置 文件 config.conf 如 以 下 
代码 所 示 。 


[glLoab 
url = "http://localhost/tp" 


name = "mobi len 


pageTitle = "Main Menu" 
bodyBgColor = #000000 
tableBgColor = #000000 
rowBgColor = #00ff00 


[db] 

host = "localhost" 
db_name = "tp" 
db_user = "root" 
db_password = "root" 
charset = "utf-8" 
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后 全 处理 你 辑 如 以 下 代码 所 示 。 


Şconfig=$this->smarty->configLoad('config.conf, 'gloab'); 
dump (Ş$config); 


configLoad 方法 是 用 于 在 后 台 获 取 配 置 文 件 信息 的 ， 如 果 在 视图 模板 中 获取 配置 信息 需要 
使 用 config_load 标签 ， 访 标签 功能 与 configLoad 方法 一 样 ， 支 持 的 属性 如 表 13-15 所 示 。 


表 13-15 config load 属性 


SS 性 说 明 S 认 值 
file 加 载 配置 文件 名 称 空 ( 必 选 ) 

section 加 载 配置 文件 节点 空 《 可 选 ) 

scope 加 载 数据 的 作用 域 ， 可 选 值 有 local, parent 和 global local 

global 加 载 变量 作用 域 false 


其 中 scope 属性 用 于 声明 数据 作用 域 ，local 表示 作用 域 为 当前 模板 : parent 表示 变量 
的 作用 域 为 当前 模板 和 当前 模板 的 父 模板 : global 表示 变量 的 作用 域 为 所 有 模板 。 通 常情 
况 下 只 需要 保持 默认 即 可 。config_load 标签 与 configLoad 方法 使 用 方式 是 一 样 的 ， 如 以 下 
代码 所 示 。 


<!—-{config_load file="confing.conf" section="gloab"}--> 

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.o0org/ 
TR/xhtml11/DTD/xhtml1-transitional.dtd"> 

<html xmlns="http://www.w3.o0org/1999/xhtm1"> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 

<title><!-—-{#pageTitle#}--></title> 

</head> 

<body bgcolor="<!-—-{#bodyBgColor#}-->"> 

</body> 

</html> 


需要 注意 的 是 ，config conf 文件 需要 ANSI 编码 格式 ， 默 认 情 况 下 不 支持 获取 中 文 节点 
以 及 中 文 配置 项 。 如 果 有 此 需要 ， 可 以 使 用 将 中 文 汉字 转 为 ASCII 编码 。 


13.7.4 ”registerPlugin〔( 注 册 插 件 ) 


在 Smarty 中 插件 是 以 函数 的 形式 进行 扩展 的 。Smarty 内 置 的 许多 标签 也 是 基于 插件 实 
现 的 ， 例 如 前 面 介 绍 的 变量 调 季 器、 判断 语句 等 。 但 这 些 插件 是 系统 内 置 的 ， 虽 然 可 以 通过 
修改 内 置 插件 的 形式 增强 Smarty 功能 ， 但 并 不 建议 这 样 做 ， 因 为 新 厂 Smarty 3 采用 了 
registerPlugin 插件 扩展 机 制 ， 允 许 开 友人 员 编 与 普通 的 PHP Gärt LI Smarty 视图 标签 ， 从 
而 实现 Smarty 高 效 扩 展 。 要 使 用 registerPlugin 注册 插件 ， 首 先 就 需要 理解 插件 的 分 类 。 
Smarty 共 分 为 4 种 类 型 的 插件 ， 即 function, block, compiler, modifiers F H EINA 
registerPlugin 方法 的 使 用 

1. registerPlugin 方法 

默认 情况 下 ，Smarty 并 不 允许 开 肥 人员 在 模板 视图 中 使 用 目 定 义 困 数 ， 这 点 与 
ThinkPHP 模板 引擎 震 别 比较 大 。 要 插入 目 定 义 函 数 ， 需 要 首先 在 后 台 使 用 registerPlugin 方 
法 注册 函数 ，registerPlusin 方法 参数 形式 如 下 。 
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public function registerPlugin($type, $tag, $callback, $cacheable = true, $cache_attr = null) 

其 中 参数 type 表示 参数 的 类 型 ， 可 选 的 值 有 function, block, compiler 及 modifier; 参 
数 tag 表示 插件 名 称 ， 该 名 称 用 于 在 模板 中 调用 ; 参数 cacheable 表示 是 否 开 局 函数 缓存 ; 2 
数 cache_attr 表示 是 合 绥 存 图 数 参数 值 。 

通常 情况 下 ， 只 需要 设置 参数 type 及 参数 tag 即 可 。 其 中 type 必须 是 一 个 合法 的 
Smarty 插件 类 型 ， 接 下 来 将 深入 介绍 Smarty 搬 件 类 型 。 

2. 插件 类 型 

插件 类 型 ， 决 定 了 插件 的 调用 形式 、 适 用 范围 、 功 能 性 质 每 。 在 前 面 的 内 容 中 ， 已 经 接 
触 过 的 插件 类 型 有 function, block 及 modifier。 接 下 来 将 重点 介绍 这 三 种 插件 类 型 。 

(1) function 函数 插件 

function 插件 是 Smarty 模板 引擎 中 最 基础 ， 也 是 最 常用 的 插件 类 型 。 顾 名 思 义 function 
插件 就 是 使 用 普通 的 PHP RREN Smarty 模板 标签 。 假 设 需要 注册 articleTop RAŽE CG ep Su 
代码 如 下 。 


function articleTop ($args, &$smarty)t 


lE (a array ($args) kk array key existasi(ititlen, Garosiil 


if (!empty ($args["title"])){ 
MPOLE TORG VUCI IOC Ern T 
Strle lesSrring: EE El imecval ($args Y lenceria i) ) 7 
Ssrr=W< ront COlor=" Y Sargs elo) .> Stiel. </rTonc>+s 


return $str; 
}else{ 

return "请 提供 文章 标题 "，; 
} 


}else{ 
return "参数 错误 "; 
} 


} 

上 述 代 码 仍 然 使 用 ThinkPHP+Smarty 开发 模式 ， 所 以 在 目 定 义 函 数 中 可 以 调用 
ThinkPHP 内 置 的 类 库 。 在 实际 应 用 开发 中 ， 访 者 还 可 以 结合 数据 库 进 一 步 处 理 数 据 。 
articleTop D Vier 2 个 参数 ， 其 中 参数 args 是 一 个 午 要 的 参数 ， 表 不 模板 传 参 
数组 ， 标 签 形式 为 {top title=" 测 试 标题 " length=10 color='"#FF0000"}。 这 就 是 说 ， 标 签 中 的 属 
性 列表 传递 给 后 台 目 定义 函数 时 ， 将 以 数组 的 方式 传递 〈 数 组 键 表 示人 参数 名 称 ， 键 值 表示 参 
数值 )。 人 参数 smarty 表示 传 入 的 Smarty 实例 对 象 ， 访 参数 不 需要 开发 人 员 传 参 。 

定义 完 目 定义 函数 之 后 ， 接 下 来 束 可 以 将 该 函数 注册 为 模板 标签 了 ， 以 便 在 模板 中 使 
D. 代码 如 下 所 不 。 


EE EE EE EE GE (tnecionY, d Jarcicleroo,)s 


Şthis->smarty->display ("index.html"); 


index.html 文件 代码 如 下 所 示 。 

<I- (top title=" Jl br Bn length- l0 color tereo000n => 
最 终生 成 的 HTML 如 以 下 代码 所 示 。 

<font color='#ER00007> 测 试 慰 题 . . .</font> 


(2) block 块 函数 插件 
block 块 函 数 插 件 相 对 于 function 函数 插件 更 利于 数据 传递 。 因 为 bock 函数 插件 是 以 成 
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对 标签 出 现 的 ， 格 式 如 下 。 


“maselo ze 0 e Olor SO 
content 


</mytag> 


如 上 述 代码 所 示 ，block 标签 必须 成 对 出 现 。 这 里 的 size 及 color 属性 将 转换 为 后 台 
mytag 函数 第 1 个 参数 〈 使 用 数组 传 参 )， 标 签 之 间 的 内 容 区 将 以 独立 的 参数 进行 赋值 

下面 将 通过 示例 代码 ， 演 示 block 函数 插件 的 使 用 。 首 先 在 common 文件 中 创建 一 个 目 
定义 函数 ， 并 命名 为 articleContent， 人 代码 如 下 所 示 。 


block 块 函数 插件 
Hacer description bere aos 


Qparam array Sparams 


大 

大 

大 

S aparam EE 
aparam oo ct o Smarty 
大 


Qparam object $repeat 


function articleContent ($args, $content, &Ş$smarty, &$repeat){ 
if (!empty ($content) && !Ş$Şrepeat){ 
$size=12; 
Le (array key exists (YsizeY, Sargs)) Ssize=Sarcgs l YsizeY] 
SSEr <ionc Slze= $size)! ee W Saros | color] s HUSU .Sooncenmne. YS SÉ 
return $str; 


} 


} 

如 上 述 目 定义 函数 代码 所 示 ， 参 数 args 表示 参数 列表 〈 数 组 类 型 ); 参数 content 表示 
block 区 块 间 内 容 ;， 参 数 smarty 表示 Smarty 实例 对 象 ， 参数 repeat 表示 是 否 重 复 解 释 。 

完成 后 ， 在 PHP 前 人 台 只 需要 将 articleContent 函数 注册 为 block 插件 即 可 ， 如 以 下 代码 
押 示 。 


HERE EE Eh El EE ER Ee e EH GEES 


Şthis->smarty->display ("index.html"); 


index.html 文件 代码 如 下 所 示 。 
EE EE EELER 
内 容 测 试 区 

EE EE 

KEE 

最 终生 成 的 HTML 如 以 下 代码 所 示 。 
<font size='10' color='#FF0000'> 

内 容 测 试 区 

</font> 

ao 


(3) modifier 变量 调节 插件 

表面 已 经 介绍 过 Smarty 内 置 的 变量 调节 堪 ， 使 用 这 些 调节 器 ， 可 以 实现 对 模板 输出 变 
量 进行 加 工 和 处 理 。 同 样 ，Smarty 也 提供 了 相应 的 变量 调节 器 扩展 机 制 ， 下 面 将 结合 示例 代 
人 码 ， 演 示 modifier 插件 函数 的 使 用 。 

Ee 

A enter description here 


EE SE DEE e e BE 
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» param lne Satar 

x @param int $length 

* Qparam string $color 

GE 

function msubstrs ($str,$star=0,Ş$length=20, $color=""){ 

if (!empty ($str))t 

import ("ORG.Util.String"); 
EE EE :meusrter (stri, neval EE 
ŞSreturn stir=Yxront COLlor='" Y, So0l6r."">Y, Sreenn str, </fFont>Y; 


return $return Str: 


}else{ 
EE EE 


} 


} 

如 上 述 代码 所 示 ， 参 数 1 ( 即 str) 是 一 个 隐 式 的 参数 值 ， 而 且 是 必 选 的 ， 表 示 变 量 本 
身 的 结果 值 ， 后 面 的 3 参数 为 自 定义 参数 ， 在 实际 应 用 开发 中 ， 读 者 可 根据 需要 添加 相应 
的 参数 。 

完成 后 ， 在 PHP 前 全 只 需要 将 msubstrs 注册 为 modifier 插件 即 可 ， 如 以 下 代码 所 示 。 


public function index (){ 
Şthis->smarty->template_dir=APP_PATH.'/Tpl/index'; 
Şarticle=M ("Article"); 


Şrows=$article->where (array ("add_user"=>"ceiba"))->select(); 


Stals- >sme Ee Dee (is, See E 
Şthis->smarty->registerPlugin ("modifier", "mstr", "msubstrs") ，; 
Şthis->smarty->display ("index.html"); 

} 


index.html FARI UN Fran, 


<!—-{foreach from=$list item=rows}--> 
11S Srows titlelmstr:0:10: red") -</li> 
<!——{/foreach}——> 


最 终生 成 的 HTML 如 以 下 代码 所 示 。 

<li><font color='red'> 库 克 就 苹果 地 图 缺陷 癌 .. .</font></1i> 
<li>-<font color=! Ped >- AE e MNOR /foOnt Se Li> 
<li><font color='red'> 眉 普 称 会 制造 足够 1P . . .</font></1i> 
<li><font color="'red'> 华 果 流 媒体 电 合 服务 过 . . .</font></1i> 
<li><font color='red'> 机 构 调 查 称 Faceb...</font></1i> 
<li><font color='red'> 李 诊 宏 称 仍 未 找到 适合 .. .</font></1i> 


13.8 Smarty 缓存 


绥 存 是 动态 网 站 开发 中 必 不 可 少 的 一 项 技术 ， 主 流 的 MVC 框架 都 针对 各 目的 模板 引擎 
提供 相应 的 缓存 模块 。Smarty 是 一 款 主流 的 PHP 模板 引擎 ， 本 身 就 内 置 了 一 套 功能 强大 、 
人 简单 易 用 的 文件 绥 存 系统 ， 接 下 来 针对 Smarty 绥 存 功能 进行 讲解 。 

13.8.1 开启 缓存 
Smarty 内 置 的 缓存 系统 是 非常 高 效 及 智能 的 ， 开 发 人 员 只 需要 开局 caching 选项 ， 即 可 


E 国 383 


PHP MVC 开发 实战 


开局 缓存 功能 ， 如 以 下 代 但 所 未 。 
// 绥 存 目录 
Şthis->smarty->cache_dir=APP_PATH. 'Runtime/'.'Cache'; 
// 绥 存 过 期 时 间 
Şthis->smarty->Ş$cache_lifetime=360000; 


$this->smarty->caching=ture; 


开启 缓存 后 ， 访 问 任 何 一 个 页 面 都 在 Cache 目录 中 生成 相应 的 静态 缓存 文件 。 但 现在 的 
模板 缓存 是 以 模板 文件 页 作为 绥 存 依据 的 ， 这 种 缓存 方式 存在 一 定 的 局 限 性 。 假 设 网 址 为 
http://tp.localhost/a.php?id=1&page=1，Smarty 将 以 aphp 作为 缓存 依据 ， 这 会 导致 用 户 在 浏 
响 第 2 页 (page=2) 时 ， 返 回 结果 与 第 1 页 相同 ， 这 显然 不 是 我 们 要 的 结 末 。 

事实 上 display 显示 方法 已 经 提供 了 相应 解释 方案 ， 开 发 人 员 只 需要 简单 设置 即 可 。 前 
面 已 经 讲述 过 display 方法 共 文 持 4 个 参数 ， 其 中 参数 2《〈 即 cache id) 束 是 用 于 解决 上 述 问 
题 的 。 如 以 下 代码 所 示 。 


Şthis->smarty->display ("index.html",md5 ($_GET["page"]));}; 


mta AA P Smarty 的 所 有 绥 存 都 是 存放 于 cache där 配置 项 指定 目录 
中 ， 并 且 以 单 级 目录 存放 。 这 就 意味 着 cache_id 需要 具备 唯一 性 ， 这 里 所 指 的 唯一 性 不 仅 针 
对 于 当前 模板 ， 而 是 针对 整个 项 目 。 假 设 网 址 为 http://tp.localhost/a.php?class_id=2&page=1， 
该 URL 并 不 是 获取 文章 ， 而 是 获取 栏目 列表 ， 如 末 继 续 使 用 $_GET["page"] 作 为 缓存 wd, Jg 
然 会 造成 缓存 冲突 。 这 时 可 以 使 用 $_SERVER[REQUEST_URT] 获 取 页 面 URL 作为 缓存 
id， 或 者 使 用 class_id+page 参数 人 的 方式 构建 缓存 ia。 总而言之 ， 需 要 确保 cache_id 值 唯 一 
性 即 可 。 

开局 绥 存 后 ， 可 以 调用 isCached 方法 检测 当前 文件 是 否 已 经 缓存 ， 如 以 下 代码 所 示 。 


public function index (){ 


Şthis->smarty->template_dir=APP_PATH.'/Tpl/index'; 
SE >See >assion( Vor" oare (Ny no ms L3 SY) ) A 
if (!Şthis->smarty->isCached("index.html",md5 (Ş$Ş_GET["page"]))){ 
// 没 有 缓存 的 情况 下 才 进 行 数据 库 得 询 操作 


Şarticle=M ("Article"); 


Şrows=$article->select(); 
SEnnis >Smarcy—>assicon(VlisE", STOwWS) 7 
} 
Şthis->smarty->display ("index.html",md5 ($_GET["page"]));}; 


} 
isCached 的 使 用 方法 与 display 方法 一 致 ， 绥 存 id 必须 确保 一 致 。 


13.8.2 局 部 缓存 


将 整个 页 和 进行 缓 企 ， 是 提高 页 面 啊 应 速度 的 一 个 重要 手段 。 但 同时 也 给 编程 灵活 性 市 
来 了 一 些 问题 ， 例 如 页 和 面 中 东 些 数据 是 实时 更 新 的 ， 这 时 如 末 系 统 读 取 的 是 缓存 中 的 数据 ， 
将 会 导致 寞 第 。 在 Smarty 中 ， 使 用 局 部 缓存 技术 能 够 避 倪 整个 页 面部 被 缓存 ， 在 一 些 PHP 
绥 存 系统 中 ， 局 部 缓存 是 一 项 非常 烦 瑛 的 操作 ， 但 在 Smarty 中 只 需要 一 对 标签 即 可 。 接 下 
来 继续 以 前 面 创建 的 例子 为 例 ， 评 细 介 绍 Smarty 局 部 缓存 的 应 用 。 
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WHAEA registerPlugin 方法 时 ， 己 经 介绍 过 该 方法 第 4 个 参数 CE cacheable) 用 于 
设置 是 否 开启 数据 缓存 ， 默 认 是 开启 的 。 利 用 该 特性 ， 只 需要 自 定义 一 个 bock 插件 ， 该 插 
件 只 需要 返回 传 入 的 内 容 《〈 即 标签 区 间 的 数据 )， 通 过 声明 禁止 缓存 数据 ， 就 达到 了 局 缓存 
的 目的 ， 步 又 如 下 。 

首先 在 函数 库 ‘common.php〉 中 创建 一 个 目 定义 块 函数 ， 并 命名 为 nocachefun， 完 成 后 
代码 如 下 所 示 。 

GE 

* Ste E 


” Enter description bere, 


x QQparam unknown type $args 

* Qparam unknown type $content 

* QPparam unknown type $smarty 

x Qparam unknown type S$repeat 

GE 

function nocachefun ($pre, $content, &Ş$smarty, &Srepeat)t 
if (!Şrepeat){ 


return $content; 


} 

接 下 来 区 可 以 在 PHP 前 台 页 面 中 注册 该 函数 了 ， 如 以 下 代码 所 示 。 

Şthis->smarty->registerPlugin("block", "nocache", "nocachefun", false); 

如 上 述 代码 所 示 ， 参 数 4 设置 为 fase 表示 禁止 绥 存 数据 。 下 向 就 可 以 在 模板 中 使 用 
nocache KZ J, ARISA Fira, 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" “nttp://www.w3.org/ 


TR/xhtml11/DTD/xhtml1-transitional.dtd"> 
<html xmlns="http://www.w3.0org/1999/xhtml1"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
<rtirctle><!=-={ pageTitle} ==></rit le> 
</head> 
<body> 
<!-—-{nocache}-->LÆHTĦE]<!--{$dtime}--><!--{/nocache}--> 
<div class="main"> 
<ul class=“ ligr "> 
<!—-{foreach from=$list item=rows}--> 
EE E A EE 
<!——{/foreach}——> 
SE 
</div> 
</body> 
</html> 


如 上 述 代 人 码 所 示 ， 位 于 nocache WREEF DEI JEE re, ARF AUER E 
多 对 nocache 标签 ， 以 实现 页 面 局 部 缓存 。 在 实际 应 用 开发 中 ， 读 者 可 根据 需要 决定 局 部 缕 
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存 的 位 置 。 


13.9 ”小结 


本 章 详细 介绍 了 Smarty 的 使 用 方法 ， 方 便 后 面 的 实战 学 习 。ThinkPHP 本 喘 承 内 置 了 一 
套 高 效 、 有 灵活 的 模板 引擎 ， 但 由 于 Smarty 在 业界 的 知名 上 度 ， 许 多 开发 人 员 仍 然 在 使 用 
Smarty 。 

本 章 一 开始 束 介 绍 了 Smarty 的 特点 ， 然 后 结合 示例 代码 详细 介绍 在 ThinkPHP 中 整合 
Smarty 的 过 程 ， 接 看 分 别 介 绍 了 Smarty 内 置 的 函数 、 标 签 等 。 

最 后 用 较 长 的 篇 幅 分 别 对 Smarty 内 置 的 3 种 目 定 义 插件 扩展 进行 了 详细 介绍 。 使 用 目 
定义 插件 ， 结 合 ThinkPHP 类 库 ， 使 得 Smarty 编程 更 加 高 效 和 灵活 。 
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第 LG 
整合 Coreseek 全 文 搜索 服务 


内 容 提要 

全 文 搜索 是 网 站 开发 中 比较 重要 的 技术 ， 用 于 解决 传统 搜索 精度 差 、 效 浴 低 、 用 户 体 验 
差 等 问题 。 在 全 文 搜 索 技 术 中 ， 常 见 的 有 数据 库 全 文 索 引 与 互联 网 搜索 引 营 站 内 搜索 。 对 于 
前 者 ， 站 憾 的 是 并 不 是 所 有 数据 库 都 能 够 提供 完善 的 支持 ， 尤 其 在 中 文 分 词 方面 对 于 后 
者 ， 由 于 基于 第 三 方 服务 ， 所 以 即时 性 及 搜索 深度 上 并 不 能 满足 所 有 网 站 的 需求 。 本 章 将 深 
入 介绍 Sphinx 全 文 搜索 引擎 ， 利 用 Sphinx 简单 的 API， 首 通 的 PHP 程序 员 也 能 开发 出 功能 
强大 、 高 效 稳定 的 全 文 搜索 引擎 。 


学 习 目 标 


了 解 全 文 搜索 概念 。 

了 解 中 文 分 词 原 理 。 

掌握 Coreseek 的 安装 。 

掌握 Coreseek 第 用 命令 工具 的 使 用 。 

掌握 Coreseek 的 索引 源 配置 。 

使 用 Sphinx API 开发 一 个 千 万 级 别 的 搜索 引擎 。 


O O 387 


PHP MVC 开发 实战 


14.1 全 文 索引 概述 


全 文 索引 是 搜索 引擎 中 最 关键 的 技术 ， 程 序 要 在 杂乱 无 章 的 文本 信息 中 提取 用 户 要 碍 找 
的 词汇 ， 如 果 采 用 传统 的 SQL 语言 进行 查询 ， 不 仅 精 度 关 ， 而 且 效率 非常 低 。 全 文 索引 的 
关键 是 分 词 技 术 ， 词 组 量 的 多 寡 决 定 了 全 文 索引 的 精度 、 效 率 、 甚 至 搜索 成 败 。 

在 全 文 搜索 技术 中 ， 主 流 的 商业 数据 库 都 提供 了 成 熟 的 全 文 索引 解决 方案 。 尘 名 的 开源 
数据 库 MySQL 从 4.x 开始 文 持 全 文 索引 。 在 开 月 全 文 索引 的 字段 中 ， 使 用 MATCH 查询 语 
人 句 即 可 实现 全 文 查 询 ， 但 是 这 种 方式 只 适用 于 英文 单词 环境 ， 对 中 文 词组 无 法 提供 文 持 。 

针对 MySQL 中 文 全 文 罕 引 ， 国 内 已 经 有 个 人 或 组 织 开 友 了 MySQL 中 文 全 文 索引 插 
件 。 例 如 海量 科技 开发 的 MySQL5.0.37--LinuxX86-Chinese+ 以 及 Hightman 论坛 开发 的 
MySQL-5.1.11-hil， 这 两 款 插件 都 能 够 为 MySQL 提供 全 文 索引 服务 ， 在 速度 及 效率 上 是 like 
语句 的 20 倍 。 

无 论 是 商业 数据 库 的 全 文 索引 ， 还 是 MySQL 全 文 索引 ， 都 是 建立 在 自身 数据 库 引 擎 上 
的 ， 在 设计 数据 表 时 ， 需 要 开发 人 员 首 先 定义 全 文 索引 字段 。 如 果 数 据 原 本 就 已 存在 ， 并 且 
创建 时 并 没有 创建 全 文 索 引 ， 这 时 再 建立 全 文 索引 将 会 给 数据 库 系 统 造成 极 大 的 影响 ， 包 括 
数据 宛 余 、 数 据 完整 性 及 系统 本 身 的 运行 效率 等 。 全 文 索引 本 质 上 是 一 个 数据 表 文 件 ， 如 果 
索引 的 内 容 过 多 ， 无 疑 会 降低 系统 的 运行 效率 ， 最 终 导 致 查询 速度 下 降 。 

本 章 介绍 的 全 文 索引 并 非 指 数据 库 全 文 索 引 ， 而 是 独立 于 数据 库 的 第 三 方 全 文 索引 服务 
器 。 目 前 最 流行 的 全 文 索引 服务 器 为 Lucene 和 Sphinx。 这 两 套 开 源 工具 都 是 出 色 的 全 文 索 
引 服 务 软件 ， 在 各 大 型 网 站 中 ， 都 有 一 定 的 应 用 规模 。 但 Lucene 是 基于 Java 实现 的 ， 所 以 
与 Java 本 身 的 融合 比较 好 。 而 Sphinx 是 基于 C 语言 开发 的 高 性 能 全 文 索 引 服务 器 ， 能 够 运 
行 于 主流 操作 平台 上 ， 它 对 单个 4GB 的 文本 字段 ， 能 够 在 军 秒 内 取得 结果 。Lucene 和 
Sphinx 的 特点 主要 如 下 。 

1. Lucene 特点 

> 索引 文件 独立 于 应 用 平台 。Lucene 定义 了 一 套 以 8 个 字 节 为 基础 的 索引 文件 格式 ， 

使 得 兼容 系统 或 者 不 同 平台 的 应 用 能 够 共享 建立 的 索引 文件 。 
> 在 传统 全 文 检索 引擎 的 倒 排 索引 的 基础 上 ， 实 现 了 分 块 索引 ， 能 够 针对 新 的 文件 建 
立 小 文件 索引 ， 提 升 索 引 速 度 。 然 后 通过 与 原 有 索引 进行 合并 ， 达 到 优化 的 目的 。 

> 优秀 的 面向 对 象 系统 架构 ， 使 得 Lucene 扩展 的 学 习 难 度 降 低 ， 方 便 扩充 新 功能 。 

> 设计 了 独立 于 语言 和 文件 格式 的 文本 分 析 接 口 ， 索 引 器 通过 接受 Token 流 完成 索引 

文件 的 创建 ， 用 户 扩展 新 的 语言 和 文件 格式 ， 只 需要 实现 文本 分 析 接 口 。 

> 默认 实现 了 一 套 强 大 的 查询 引擎 ， 用 户 无 需 编写 任何 代 人 码 即 可 获得 强大 的 查询 功 

能 ，Lucene 的 查询 引擎 默认 实现 了 布尔 操作 、 模 糊 查 询 、 分 组 查询 等 。 

2. Sphinx 特点 

> LRE], PJA 10MB/s, m Lucene 建立 索引 的 速度 是 1.8MB/s。 

> 局 扩展 性 (实测 最 高 可 对 100GB DVC Ars, PRIER 1 亿 条 记录 )。 

> 文 持 分 布 式 检索 。 
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> 支持 基于 短语 和 基于 统计 的 复合 结果 排序 机 制 |。 

> 文 持 任意 数量 的 文本 字段 (数值 属性 或 全 文 检索 属性 )。 

> 文 持 不 同 的 搜索 模式 ， 例 如 完全 匹配 、 短 语 匹 配 、 任 一 匹配 等 。 

> 文 持 作 为 MySQL 的 全 文 存储 引擎 (SphinxSE)。 

> 官方 提供 主流 开发 语言 API， 包 括 PHP, Java, CH CH. 

这 两 套 全 文 索引 引擎 ， 各 有 特点 。 绽 合 来 看 Lucene 功能 比较 多 ， 例 如 内 置 查 询 语言 ， 
并 且 文 持 多 种 查询 方式 ，Sphinx 最 显著 的 特点 是 它 本 喘 不 储存 数据 ， 上 只 保存 索引 id, MUA 
询 效率 非常 突出 。 本 章 将 重点 介绍 Sphinx 的 实战 应 用 。 


14.2 Coreseek 基础 


与 其 他 国外 全 文 索引 引擎 一 样 ，Sphinx 本 身 并 没有 提供 中 文 分 词 功能 。Coreseek 的 出 
现 ， 很 好 地 解决 了 上 述 问题 ， 接 下 来 将 对 Coreseek 进行 介绍 。 


14.2.1 Coreseek 概述 


Coreseek 是 一 球 基 于 Sphinx 独立 发 行 的 全 文 搜索 引擎 ， 能 够 在 极 少 的 时 间 内 对 海量 的 
中 文 数据 创建 案 引 库 。 基 于 Sphinx 强大 及 融 效 的 搜索 服务 ， 能 够 为 中 文 搜索 环境 提供 出 色 
的 用 户 体 验 。 与 Sphinx 一 样 ， 虽 然 称 之 为 搜索 引擎 ， 但 事实 上 Coreseek 并 不 存放 原文 数 
据 。 对 开发 者 而 言 ， 数 据 存 放 过 程 是 透明 的 ， 基 于 海量 的 中 文 词 组 库 ，Coreseek 能 够 对 常见 
的 中 文 词汇 建立 索引 ， 并 生成 唯一 索引 号。 开发 人 员 通 过 索引 写 ， 结 合 SQL 语句 查询 到 最 
终 的 原文 数据 。 整 个 流程 如 图 14-1 所 示 。 

通过 图 14-1 可 以 看 到 ，Coreseek 在 创建 中 文案 引 时 ， 是 根据 MMSEG 来 创建 的 。 首 
先 ， 系 统管 理 员 配 置 好 数据 源 豫 动 后 ，indexer 索引 程序 将 载 入 MMSEG 词 库 ， 并 连接 到 数 
据 源 。 如 果 数 据 源 全 文 包 含有 MMSEG 任何 一 条 词 条 ，Sphinx 将 对 该 原文 创建 索引 并 保存 到 
内 置 的 key-value 数据 库 中 (key 存放 词 条 ，value 存放 原文 id)。 这 就 意味 着 用 户 搜 索 到 
Sphinx 数据 库 中 的 词组 ， 束 可 以 根据 Sphinx 返回 的 id， 得 到 数据 原文 。 

整个 用 户 搜 索 过 程 ，Sphinx 并 没有 与 数据 源 进行 了 互动， 用户 搜索 的 只 是 Sphinx 数据 
Æ. HF Sphinx 数据 库 只 存放 key-value 数据 ， 所 以 啊 应 速度 非常 快 。 最 后 通过 曲线 获取 数 
据 源 原文 ，Sphinx 相当 于 得 询 中 间 件 ， 从 而 很 好 地 解决 了 效率 、 安 全 、 精 度 的 问题 。 

这 其 中 直接 决定 搜索 精度 的 是 MMSEG 中 文 词 条 记录 《〈 词 条 越 多 精度 越 遍 )。 对 于 英文 
检索 而 言 ， 由 于 英文 词 条 是 有 衬 格 的 ， 所 以 不 需要 额外 的 词 库 。 但 对 中 文 环境 而 言 需要 使 用 
特定 的 词 条 分 割 技 术 ， 把 中 文 词 条 提取 出 来 。MMSEG 就 是 一 套 高 效 的 中 文 词 条 分 割 类 库 ， 
结合 Sphinx 就 能 够 实现 将 分 割 的 词 条 收录 到 内 置 的 数据 库 中 ， 最 终 用 户 人 查询 的 也 是 Sphinx 
内 置 的 数据 库 。 

傈 而 言 之 ， 全 文 搜索 主要 分 两 个 步 又 : 首先 程序 在 Sphinx 数据 库 中 搜索 词 条 ;， 然后 得 
到 词 条 id; 最 后 根据 返回 的 id 使 用 SQL 进行 查询 。 在 这 个 过 程 中 ， 用 户 搜 索 Sphinx 数据 库 
时 ，MMSEG 分 割 模式 的 更 改 ， 将 和 直接 影 啊 搜索 的 精度 。 在 接 下 来 的 内 容 中 将 结合 示例 进行 
介绍 。 下 和 耐 站 先 需 要 安装 Coreseek。 
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Sphinx 配置 文件 WE 
MySQL 


计划 任务 或 手动 启动 ndexer 主 索引 进程 MSSQL 


d PostgreSQL 


MMSG 中 文 分 词 创建 索引 Xmlpipe2 
根据 MMSG 中 文 
词 条 建立 索引 lid 


Di AS TOS 


返回 搜索 词组 Sage 


计划 任务 或 手动 启动 searchd 守护 进程 


搜索 API 
返回 原文 记录 
Java, PHP, CH 一 一 一 一 一 一 一 一 一 一 一 一 一 d 


搜索 网 页 界面 


图 14-1 Coreseek 全 文 索引 过 程 


14.2.2 ”在 Windows 下 安装 Coreseek 


Coreseek 能 够 运行 在 Windows, FreeBSD, Mac OS 以 及 Linux 等 主流 操作 系统 上 。 与 多 
数 开源 软件 一 样 ，Coreseek 虽然 能 够 运行 在 Windows 操作 系统 上 ， 但 通 钊 只 用 于 开发 阶段 的 
测试 及 学 习 ， 如 果 需 要 部 普 到 生产 环境 ， 应 该 选择 Linux 及 FreeBSD。 为 了 方便 学 习 及 测 
试 ， 下 面 首 先 介 绍 Windows 下 安装 Coreseek 的 过 程 。 

1. 下 载 安装 

在 安装 Coreseek 前 ， 首 先 需 要 安装 Microsoft Visual C/C++ 编 详 环境 。 斌 者 可 以 在 
http://Wwww.coreseek.cn/news/7/52/ 网 页 找到 Coreseek for Windows 安装 包 ， 这 里 将 下 载 
Coreseek-3.2.14-win32.zip。 下 载 完成 后 ， 将 得 到 Coreseek 类 库 及 API WRI. HRALA 


构 如 下 。 
coreseek-3.2.13-win32 
api/ ee 第 三 方 应 用 程序 API 目录 
bin/ weeeeee Sphinx 运行 文件 目录 
Ee mi sphinx 配置 文件 目录 
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test .cmad teese Sphinx 测试 工具 
yar/ eeeeee Sphinx 数据 库 
README.txt onsen Coreseek 说 明文 档 


在 解压 文件 时 ， 尽 量 避 免 解 压 到 中 文 名 称 目 录 。 接 下 来 只 需要 简单 配置 就 可 以 测试 
Coreseek J- 


2. 配置 文件 
进入 etc 目录， 将 会 看 到 Coreseek 默认 已 经 提供 了 5 个 配置 文件 ，etc 目录 结构 如 下 。 
etc 
pysource "onsen python $% OWARE 
Csft.conf eseese 配置 XML 数据 源 
csft demo python.conf pp 配置 python 接口 〈 万 能 接口 ， 需 要 安 钱 ActiveState Python) 
csft demo python pymssql.conf ……………: 配置 MSSQL 数据 源 
csft mysql.conf seee. HE MySQL 数据 源 
mmseg. ini ee MMSEG 配置 文件 
unigram.txt eese. MMSEG 词 库 


为 了 方便 测试 ， 这 里 只 需要 配置 csft mysqlconf 文件 即 可 。 在 配置 该 文件 前 ， 首 先 需 要 
MIRAA MySQL 数据 库 ， 并 且 能 够 正常 访问。 打开 csft mysql.conf 配置 文 
件 ， 这 里 将 对 tp 数据 库 中 的 tpk article 数据 表 创 建 全 文案 引 ， 代 人 码 如 下 。 

# 源 定义 

source mysql 


{ 


type = mysql 

sal NOST = localhost 

sql user = FOOT 

sal pass = root 

sql db = tp 

SG Port = 3306 

sql query pre = SET NAMES utf8 

# 关 闭 查 询 SOL 缓存 

sql query pre = SET SESSION Query cache CypPe=SOrE 


AS Olmusoal ANE A 
sql query = SELECT ic, title, content FROM tpk article 


# 对 排序 字段 进行 注释 

#sdl atte uint = add time 

Hoo acte emen tame = UNIX TIMESTAMP (aco time) AS addec. ts 
# 仅 被 命令 行 搜索 所 用 ， 用 来 获取 和 显示 文档 信息 

SQL query info 三 本 SDDS 


} 

#index 定 义 

index mysql 

l 

# 对 应 的 source 名 称 


source = mysql 
path = var/data/mysql 
cdocinto = extern 
mlock = 0 
morphology nome 
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ma word Llen = 
mem] strie = 0 
#charset dictpath = /usr/local/mmseg3/etc/ 


Charset Cicteathn = 
charset type = 


} 
teja index EX 
indexer 


{ 


SG/ 
zA REES 


mem limit = 128M 
} 
#searchd 服务 定义 
searchd 
{ 
listen = 9312 
reac timeout 三 5 
max children = 10 
max matches = OOOO 
seamless rotate = 
preopen indexes = 0 
unlimnk olal = 
Did eile var/lLoo/searchd mysg pid 
log = var/log/searchd mysql.log 
query log = var/log/query mysql.log 
DARRERE 个 简单 的 Coreseek 配置 文件 ， 


title, content 字段 创建 全 文 索引 。 


要 人 简单 了 解 即 可 。 
3. 局 动 测试 


更 加 详细 及 高 级 的 配置 将 在 后 面 进行 介绍 


该 配置 将 对 tpk article 数据 表 中 的 
在 此 读者 只 需 


Wad o 以 进行 测试 了 。Windows 平台 下 的 Coreseek 测试 比较 简 


单 ， 首先 打开 命令 Sí 


Zimo AYFER] Coreseek 产 代 包月 录 ， 如 图 14-2 所 示 。 


m EER: 命令 提示 符 


图 14-2 切换 到 Coreseek HIEL Ha 


ZATRE Coreseek 配置 文件 (默认 为 csft.conf)。 输 入 “bin\indexer -c etc\csft mysql. 


conf” 
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命令 ， 如 图 14-3 所 示 。 


marassa E Zë 


D:\E-disk\coreseek-3.2.13-win32\coreseek-3.2.13-win32>bin\indexer -c etc\csft_mysql.conf 


Coreseek Fulltext 3.2 [ Sphinx 0.9.9-release (r2117)] 
Copyright (c) 2007-2010, 
Beijing Choice Software Technologies Inc (http://www.coreseek.com) 


using config file ‘etc\csft_mysql.conf'... 
total 0 reads, 0.000 sec, 0.0 kb/call avg, 0.0 msec/call avg 
total 0 writes, 0.000 sec, 0.0 kb/call avg, 0.0 msec/call avg 


D:\E-disk\coreseek-3.2.13-win32\coreseek-3.2.13-Win32> m 


K 14-3 45 


定 Coreseek 配置 文件 
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如 果 配 置 文件 无 误 ， 接 下 来 就 可 以 使 用 indexer 工具 创建 全 文 索引 数据 库 了 。 输 入 
“bin\indexer -c etc\csft mysqlconf --all ”命令 ， 如 图 14-4 所 示 。 


D 管理 员 : 命令 提示 符 


D:\E-disk\coreseek-3.2.13-win32\coreseek-3.2.13-win32>bin\indexer -c etcNcsft_musql .conf --a 
Coreseek Fulltext 3.2 [ Sphinx 0.9.9-release (r2117)] 

Copyright (c) 2007-2010, 

Beijing Choice Software Technologies Inc (http://www .coreseek .com) 


using config file ‘etc\csft_mysql.conf'... 


indexing index ‘mysq 
collected 6 docs, 0.0 MB 

sorted 0.0 Mhits, 100.0% done 

total 6 docs, 11867 bytes 

total 0.074 sec, 160211 bytes/sec, 31.00 docs/sec 

total 1 reads, 0.000 sec, 13.0 kbycall avg, 0.0 msec/call avg 
total 5 writes, 0.000 sec, 6.6 kbycall aug, 0.1 msec/call aug 


图 14-4 创建 全 文 索引 
如 图 14-4 所 示 ，indexer 创建 完 索 引 后 ， 会 清楚 地 显示 索引 条 数 、 数 据 大 小 等 。 这 在 大 
型 数据 中 是 非常 必要 的 。 完 成 前 面 的 步 又 ， 接 下 来 束 可 以 使 用 命令 工具 进行 简单 的 测试 了 。 
在 测试 数据 前 ， 首 先 观察 tpk article 数据 表 记 录 ， 如 图 14-5 所 示 。 


-T> id add time title content 
O 2 X 1 13464016% 库 克 就 苹果 地 图 缺陷 向 用 户 道 歉 : 未 提供 一 流体 验 网易 科技 讯 9 月 28 日 晚间 消息 ， 苹 果 CEO 蒂 姆 . 库 克 今 日 就 苹果 地 图 存在 诸多 功能 问题 向 用 户 发 函 致 菊 ， 称 .… 
A X 2 1346401704 索尼 投资 6.44 亿 美元 成 为 奥 林 巴 斯 第 一 大 股东 新 华 网 东京 9 月 28 日 电 ( 记者 BES ) 日 本 奥 林 巴 斯 光学 公司 28 日 宣布 与 索尼 结 成 资本 与 业务 合作 关系 。 索 .… 
O] 2 X 3 1346401694 夏普 称 会 制造 足够 iPhone5 所 需 的 显示 屏 网 易 科 技 讯 9 月 28 日 消息 ， 据 国外 媒体 报道 ， 夏 普 已 经 制造 了 足够 用 于 苹果 iPhone5 的 旦 示 屏 。 这 也 雪 .… 
门 A X 4 1346401704 ， 芋 果 流 媒体 电台 服务 遇 障 碍 : 索尼 /ATV 对 价格 不 满 ” 网 易 科技 讯 9 月 29 日 消息 ， 据 国外 媒体 报道 ， 苹果 的 互联 网 电台 服务 计划 最 近 在 获得 由 索尼 /ATV 控 制 的 音 ... 
O 2 X 5 1346401694 机 构 调 查 称 Facebook 中 国 用 户 数 超 6000 万 网 易 科 技 讯 9 月 29 日 消息 ， 据 国外 媒体 报道 ， 英 国 市 场 调查 机 构 GlobalWebIndex 近 日 公布 了 一 … 
门 万 X 6 1346401704 李彦宏 称 仍 未 找到 适合 移动 互联 网 的 商业 模式 网 易 科 技 讯 9 月 29 日 消息 ， 据 国外 媒体 报道 ， 百 度 创 始 人 、 董 事 长 兼 首席 执行 官 李彦宏 昨天 在 斯 坦 福 大 学 举行 … 


图 14-5 tpk article 数据 表 记 录 


假设 需要 搜索 包含 关键 字 为 “28” 的 记录 ， 那 么 只 需要 输入 “bin\search -c ete\csft_ 
mysql.conf 28” 命 令 即 可 ， 搜 索 结果 如 图 14-6 fra, 


en 管理 员 : 命令 提示 符 
category: 
Grenz 


add_user:ceibā 
add_time=1346401694 


Lock uersionz 


': 3 documents, 4 hits 


图 14-6 搜索 测试 


可 见 搜索 的 结 末 与 我 们 期 竺 的 结果 是 一 致 的 ， 即 一 共有 3 条 包含 关键 字 为 “28” 的 文 
草 记 录 。 通 过 前 面 的 测试 ， 可 以 看 出 Coreseek 搜索 精度 是 非常 高 的 。 由 于 在 命令 行 下 
search 工具 对 中 文 编码 并 不 能 正确 识别 ， 所 以 这 里 并 没有 测试 中 文字 符 ， 事 实 上 Coreseek 
已 经 能 够 提供 中 文 检索 了 ， 后 和 面 将 使 用 PHP 代码 测试 中 文 词组 检索 。 

表面 已 经 提 到 过 ， 在 Windows 平台 下 的 Coreseek 主要 用 于 开发 阶段 的 测试 。 所 以 在 安 
IT Coreseek 时 并 没有 日 动 创建 启动 服务 。 这 里 只 需要 使 用 “bin\searchd.exe -c etc\csft 
mysql.conf” 命 令 启 动 Coreseek 即 可 ， 如 图 14-7 所 示 。 
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Di 管理 员 : 命令 提示 符 - bin\searchd.exe -c etc\csft_mysql.conf 


D:\E-disk\coreseek-3.2.13-win32\coreseek-3.2.13-win32>bin\searchd.exe -c etc\csft_mysql .conf 
Coreseek Fulltext 3.2 [ Sphinx 0.9.9-release (r2117)] 

Copyright (c) 2007-2010, 

Beijing Choice Software Technologies Inc (http://www .coreseek.com) 


WARNING: forcing --console mode on Windows 
using config file 'ʻetc\csft_mysql.conf'... 
listening on all interfaces, port=9312 
accepting connections 


图 14-7 启动 Coreseek 主 进 程 


至 此 Coreseek WER J o EAn LE sce 命令 将 Coreseek 加 入 到 Windows 启动 
服务 中 ， 避 人 免 需 要 挂 起 命令 终端 ， 影 啊 开 发 效率 。 合 令 如 下 。 


se Creare CoreseekServer binoathns= "D: \E-dlisk\coreseek- 3.2.13-win32\bin\searched, exe 
=@ D; \E-disk\coreseek-3.2.13-win32\etc\csft mysgl.coni" type= share start= auto 
displayname= Cor -seek TX ezko denen: Bpcäs/kveentävatem 


读者 可 根据 Coreseek 实际 路 径 ， 修 改 相应 的 binpath 28 "9 HI of. AHRI, ERSE 
理 面 板 中 ， 可 以 找到 前 面 创建 的 CoreseekServer 服务 ， 如 图 14-8 rz, 


Coreseek 全 文 搜索 服务 名 称 ex ”状态 Kaes 登录 为 < 
2 Bluetooth Support Service Blue.. CR. Za 本 地 服 … 
启动 此 服务 & Bonjour 服务 让 硬 .… 手动 FER... z 
Z BranchCache 此 服 .…… 手动 网 络 服 .… 
CG c20ukdrwsvc 手动 本 地 系 … 
Z CDROM_Detect 手动 本 地 系 … 
Š% Certificate Propagation HA.. BE.. F 本 地 系 … 
k CNG Key Isolation CN... GJA... 手动 本 地 系 .… 
ZÆ CNTV CBox Service CNT.… 手动 本 地 系 … 
Z COM + Event System 支持 ..， 已 启 ..。 手 动 本 地 服 .… 
2 COM: System Application EE... 手动 本 地 系 ..… 
Š Computer Browser 维护 ..， 已 启 ..。 手动 本 地 系 … 
Coreseek 全 文 搜索 服务 
各 Credential Manager 为 用 .… 手动 FER... 
Æ Cryptographic Services 提供 ..， 已 启 ..， 自动 Lob: S 


图 14-8 CoresseekServer 服务 


14.2.3 在 Linux 下 安装 Coreseek 


在 Windows 下 的 安装 过 程 ， 只 能 用 于 测试 及 调试 ， 应 该 尽量 避免 在 生产 环境 中 使 用 。 
接 下 来 将 在 Cent OS 6.0 操作 系统 中 安装 Coreseek 3.2.14， 安 装 完 成 后 可 以 用 于 测试 环境 ， 也 
可 以 用 于 生产 环境 ， 步 又 分 别 如 下 。 

1. 安装 Coreseek 3.2.14 

自 完 升级 或 安装 系统 依赖 库 。 


[root@~]# yum install make gcc g++ gcc-c++ libtool autoconf automake imake mysql- 


devel libxml2-devel expat-devel 


操作 完成 后 ， 接 下 来 下 载 Coreseek 3.2.14. 


[root@~]# mkdir -p /datal 
[root@~]# cd /datal 


[root@~]# wget http://www.coreseek.cn/uploads/csft/3.2/coreseek-3.2.14.tar.gz 


[root@~]# tar -zxvf coreseek-3.2.14.tar.gz 


完成 上 述 操作 后 ， 将 得 到 Coreseek 3.2.14 源码 文件 。 通 过 前 面 的 操作 ， 现 在 束 可 以 安装 
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Coreseek 3.2.14 了 。 在 安 猴 表 首先 需要 编 详 MMSEG 插件 。 


[root@~]# cd coreseek-3.2.14 


[root@~]# cd mmseg-3.2.14/ 


[root@~]# ./configure --prefix=/usr/local/mmseg3 
config.status: creating Makefile 
config.status: WARNING: 'Makefile.in' seems to ignore the --datarootdir setting 


config.status: error: cannot find input file: src/Makefile.in 
配置 MMSEG 插件 时 ， 将 出 现 Makefile 错误 。 可 以 通过 安装 autoconf 和 automake 解 
决 。 下 和 耐 是 解决 过 程 。 


[root@linux mmseg-3.2.14]# yum -y install autoconf automake 

[root@linux mmseg-3.2.14]# aclocal 

aclogal:conrfigure.in:20: warning: macro TAM PROG LIBTOUL™ not found in library 
[root@linux mmseg-3.2.14]# yum -y install libtool 


[root@linux mmseg-3.2.14]# aclocal 


[root@linux mmseg-3.2.14]# libtoolize --force 


Putting files in AC CONFIG AUX DIR, 'config' 


qd- 


[root@linux mmseg-3.2.14]# automake --add-missing 


qd- 


[root@linux mmseg-3.2.14]# autoconf 


qd- 


[root@linux mmseg-3.2.14]# autoheader 


[root@linux mmseg-3.2.14]# make clean 


TREI, Hai E e ABN H o 
[root@linux mmseg-3.2.14]# ./configure --prefix=/usr/local/mmseg3 
Configuration: 


Source code location: 


Compiler: gcc 

Compiler flags: -g -02 

Host System Type: i686-redhat-linux-gnu 
Install path: /usr/local/mmseg3 


See config.h for further configuration information. 


完成 前 面 的 步骤 后 ， 接 下 来 就 可 以 安装 MMSEG 了 。 


[root@linux mmseg-3.2.14]# make && make install 


test -z "/usr/local/mmseg3/etc" || /bin/mkdir -p "/usr/local/mmseg3/etc" 
/usr/bin/install -c data/unigram.txt data/uni.lib data/mmseg.ini '/usr/local/ 
mmseg3/etc' 

make[2]: Leaving directory '/home/data/coreseek-3.2.14/mmseg-3.2.14' 


make[1]: Leaving directory '/home/data/coreseek-3.2.14/mmseg-3.2.14' 


configure 可 用 的 配置 项 如 下 《可 通过 一 help 选项 获得 详细 的 帮助 信息 ) 
O O 395 


PHP MVC 开发 实战 


--prefix: 定义 Coreseek 安装 路 和 任 ， 比 如 --prefix=/usr/local/sphinx。 

--with-mysql: 当 检 测 到 不 存在 MySQL 头 文 件 和 库 文 件 时 ， 手 动 指 定 存放 路 径 。 
--with-pgsql: 当 检 测 到 不 存在 PostgreSQL 头 文 件 和 库 文件 时 ， 手 动 指定 存放 路 径 。 
--with-mmseg: 局 用 MMSeg 中 文 分 词 插 件 ， 并 手动 声明 MMSeg 头 文 件 和 库 文 件 
路 径 。 

> --with-python: 局 用 Python 数据 源 驱 动 文 持 ( 需 要 预 完 安装 Python 2.6). 

如 果 在 安装 MMSEG 过 程 中 出 现 如 下 信息 。 


/usr/local/sphinx-0.9.9/src/sphinx.cpp:20060: undefined reference to 'libiconv_ 


VW vV V 


open’ 

有 效 的 解决 方法 如 下 。 

首先 打开 configure 文件 ， 找 到 “#define USE LIBICONV 1”， 去 掉 “#” 注 释 符 ， 并 将 
1 改 成 0。 然 后 保存 configure 文件 ， 香 独 编 译 并 安装 即 可 这 种 解决 方法 也 适用 其 他 软件 安 
Ze EE, 

MMSEG FP L ZREN, F MRa Do Coreseek 3.2.14 了 ， 过 程 如 下 。 
[root@linux mmseg-3.2.14]# ln -s /usr/local/mmseg3/bin/mmseg /bin/mmseg 


[root@linux mmseg-3.2.14]# cd .. 


[root@linux coreseek-3.2.14]# cd csft-3.2.14/ 


[root@linux coreseek-3.2.14]# ./configure --prefix=/usr/local/coreseek --without- 
unixodbc --with-mmseg-includes=/usr/local/mmseg3/include/mmseg/ --with-mmseg- 
libs=/usr/local/mmseg3/lib/ --with-mysql 
[root@linux coreseek-3.2.14]# make & make install 
mA MAA ER, MMSEG 及 Coreseek 3.2.14 W ZIET J o W ls 命令 可 以 查看 到 安 

装 后 的 目录 及 文件 。 


[root@linux coreseek-3.2.14]# ls /usr/local/coreseek/ 


bin etc var 

[root@linux coreseek-3.2.14]# 

2. 测试 Coreseek 中 文 分 词 

在 Linux 环境 下 ，Coreseek 安 猜 完成 后 ， 提 供 了 一 个 example.sql 文件 ， 该 文件 是 一 个 标 
准 的 MySQL 和 奉 询 文件 ， 用 于 建立 测 斌 数据库。 同时， 为 了 方便 测试 MMSEG 插件 ， 
Coreseek 安装 包 还 提供 了 一 个 test.xml 测试 文件 及 testpack 测试 包 。 接 下 来 将 使 用 testpack W 
试 工具 及 配置 文件 ， 对 MMSEG 进行 测试 ， 步 骤 如 下 。 

[root@~]# cat /datal/coreseek-3.2.14/testpack/var/test/test.xml 

运行 上 述 命令 后 ， 将 会 显示 一 篇 基于 Xmlpipe2 数据 格式 的 测试 文章 ， 如 图 14-9 所 示 。 

接 下 来 将 使 用 MMSEG 创建 中 文 词 条 ， 并 保存 到 Sphinx 数据 库 中 。 步 骤 如 下 。 


[root@~]# cd /datal/coreseek-3.2.14/testpack/ 


[root@~]# /usr/local/mmseg3/bin/mmseg -d /usr/local/mmseg3/etc var/test/test.xml 


[root@~]# /usr/local/coreseek/bin/indexer -c etc/csft.conf --all 
通过 前 面 的 步骤 ， 接 下 来 就 可 以 搜索 test.xml 文件 中 的 数据 了 。 假 设 需要 搜索 内 容 包含 
“网 络 搜索 ”的 数据 ， 步 又 如 下 。 
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ocuments. sq r 
[root@bogon ace Ts wl ET GEES SR, Ae eebe Xml 
/datal/coreseek-3.2.14/testpack/var /test/test 
[root@bogon etc]# cat /daral/coreseek- 3.2.14/testpack/var/test/test. ml 
<?xml] version="1. 0" encoding="utf-8"?> 
<sphinx:docset> 
<sphinx:schema> 
<sphinx:field name="subject"/> 
<sphinx: field name 一 "Content" Ss 
<sphinx:attr name="published" type="timestamp"/> 
<sphinx:attr name="author_id" type="int" bits="16" default="1"/> 
</S hinx : schema> 
<sphinx:document id="1"> 
a SS BLETE E Si subject> 
<published>1270131607</published> 
<Content> 据 国外 媒体 报道 ， ONET, Zb Je NUL mn. GEE E A T A 
谍报 道 称 ， 目 前 谷歌 与 百度 已 经 达成 了 收购 协 将 择机 对 外 公布 。 百 度 的 管理 层 将 100% 保 锁 ， 但 会 将 项 目 缩减 ， 包括 有 啊 商城 
EE Ee Beta Gëf E E A Youtube RIE. (YouTube 在 大 陆 区 因 内 容 审 查 暂 
“a ja 
CB 息 似 于 得 到 了 谷歌 CE0 施 密 特 的 确认 ; 在 其 twitter 上 用 简短 而 蚂 昧 的 文字 进行 了 表述 : ”Withdraw from that market? u'll 
also see another result, just wait... ”意思 是 : 从 那个 市 场 运 出 ?你 还 会 看 到 另外 一 个 结果 。 这 无 疑问 ， 那 个 市 场 指 的 就 是 中 
国 大 陆 。 而 另外 的 结果 ， 对 应 此 媒体 报道 ， 就 是 收购 百度 ， 从 而 曲线 返回 大 陆 搜索 市 场 。 
在 最 近 卫 了 刚 结束 的 深圳 IT 领袖 峰会 上 ， 李 彦 安 曾 言 ， “谷歌 没有 退出 中 国 ， RE 在 香港 ”。 也 似乎 在 验证 被 收购 的 这 一 事实 。 
截止 发 稿 ， 度 的 WEE 市 值 为 207 忆 美元 。 人 和 谷歌 以 高 达 300 亿 美元 的 价格 ， 实际 洲 价 高 达 50%.。 而 谷歌 市 值 高 达 1796 亿 


ZS , 作 这 样 的 决策 之 中 。 
SE CEO ge pop kp, — tr Re Lrign/, ege SI DICH, St Em EH E 28 
i 首 ; e Si 化 战 战略 中 ， 拒 绾 采用 海外 并 购 的 方式 ， 而 是 架 到 了 从 日 本 市 场 开 始 的 海 


图 14-9 Coreseek 测试 文章 


[root@bogon testpack]# /usr/local/coreseek/bin/search -c etc/csft.conf 网 络 搜索 
Coreseek Fulltext 3.2 [ Sphinx 0.9.9-release (2117) 
Copyright (c) 2007-2011, 


Beijing Choice Software Technologies Inc (http://www.coreseek.com) 


using config file 'etc/csft.conf'... 


index 'sml': query MANA ': returned 1 matches of 1 total in .0.141 sec 


displaying matches: 


1. document=1, weight=1, published=Thu Apr 1 22:20:07 2010, author id=1 


words: 

1. ' 网 络 ': 1 documents, 1 hits 

2. ' 搜 索 ': 2 documents, 5 hits 

从 结果 中 可 以 看 到 ，MMSEG 将 输入 的 “网 络 搜索 ”一 词 拆 分 成 了 “网 络 ” 和 “搜索 ” 
两 个 词 条 ， 这 就 意味 着 ， 全 文 数据 中 无 论 是 存在 “网 络 搜索 ”还 是 只 包含 “网 络 ” 或 “ 搜 
索 ” 都 能 够 被 Coreseek 检索 。 读 者 可 以 修改 test.xml 数据 ， 观 察 MMSEG 中 文 分 词 的 变化 。 


14.3 ”Coreseek 管理 工具 


与 Windows 有 版 的 Coreseek 一 样 ， 在 Linux 版 本 中 同样 提供 了 一 系列 管理 工具 ， 包 括 
indexer、search、searchd 等 。 下 和 面 分 别 介 绍 这 些 工 具 的 作用 及 使 用 方式 。 


14.3.1 indexer 


indexer 工具 是 Sphinx 站 站 的 一 个 重要 的 节令 生理 工具 ， 用 于 为 系统 创建 案 引 文档 。 
indexer 是 一 个 统一 的 管理 工具 ， 无 论 数据 源 来 自 于 数据 库 ， 还 是 XML，indexer 工具 的 使 用 
a a 所 以 无 论 是 用 于 命令 行 ， 还 是 作为 计划 任务 脚本 ， 数 据 源 被 修改 后 ， 都 不 
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会 影响 indexer 的 执行 。indexer 工具 命令 格式 如 下 。 

indexer [OPTIONS] [indexnamel [indexname2 [...]]] 

indexer MAE T IERI WERAK KRW., AE MKX) indexer 工 其 的 沼 
用 选项 进行 介绍 。 

(1) --config 配置 文件 

Coreseek 安装 完成 后 ， 默 认 sphinx.confdist、sphinx-min.conf det 作为 配置 文件 模板 “〈 注 
mt Windows 有 版 本 之 间 的 区 别 )。 假 设 Coreseek 安装 在 /asrvlocalMcoreseelv H at, A 
sphinx.conf 配置 文件 的 完整 路 径 束 是 /usr/local/coreseek/sphinx.conf。 如 果 sphinx.conf 文件 被 
删除 或 改名 ，indexer 工具 会 尝试 在 当前 目录 中 查找 sphinx.conf 文件 。 当 然 ， 开 发 人 员 也 可 
以 通过 --config 选项 强制 指定 indexer 配置 文件 位 置 ， 如 以 下 代码 所 示 。 

[root@~]# indexer --config /usr/local/coreseek/sphinx.conf myindex 

值得 需要 注意 的 是 ，Coreseek 默认 的 配置 文件 为 csft.conf。 为 了 方便 演示 ， 这 里 将 使 用 
sphinx-min.conf.dist 模板 作为 Coreseek 配置 文件 ， 只 需要 将 该 文件 复制 为 esft.conf 即 可 。 

[root@bogon bin]# cp /usr/local/coreseek/etc/sphinx-min.conf.dist csft.conf 

一 个 简单 的 Coreseek 配置 文件 至 少 需要 由 source、index、indexer、searchd 四 大 部 分 组 


成 。 如 以 下 代码 所 示 。 
# 配 置 数据 源 


source main 


{ 


} 
# 生 成 索引 


index main 


{ 


} 
# 配 置 索 引进 程 参数 


indexer 

{ 

} 

# 配 置 搜 索 进程 参数 
searchd 


{ 
} 


为 了 方便 测试 ， 这 里 将 修改 默认 的 csft.conf 配置 文件 ， 帮 助 读 者 加 深 对 Coreseek 配置 文 
件 的 认识 ， 代 人 码 如 下 所 示 。 

# 

i Minimal Sphinx Configuration samole (clean, simple, functional) 


# 
# 主 索引 


source maim 


{ 


Tye = mysql 

Ee EE 

sql user = POOT 

sql pass = FOOT 

sgl cs = TEST 

A Pore = 53506 i Optional, coefault is 3306 
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sal uery pre SET SESSION query EE EE 


SET NAMES UTEG 


sal Guery pre 


saql query = SELECT 1C title, content PROM tpk article 
saql attr uing = 人 GG time 

sql attr timestamp = id 

sql query infe 三 ae EROM t E Rees 


} 


1noex maLm 


source = reci 
path = /usr/local/coreseek/var/data/main 
cocinfo = extern 
Charset Type = Sbcs 
} 
indexer 
{ 
mem Limit = 32M 
} 
searchd 
{ 
port = 9312 
log = /usr/local/coreseek/var/log/searchd. log 
query log = /usr/local/coreseek/var/log/query.log 
Lead tlimaouEe >55 
max Chileren = 30 
pid file = /usr/local/coreseek/var/log/searchd.pid 
ma matches = 1000 
seamless rotate = 1 
preopen lnoexes = 0 


unlink old = | 


} 

上 述 配 置 文 件 与 前 面 介绍 的 Windows 版 Coreseek 配置 文件 相 类 似 ， 只 有 searchd 配置 存 
在 微小 区 别 。 这 样 一 个 简单 的 Coreseek 配置 文件 就 创建 完成 了 ， 接 下 来 的 演示 示例 均 使 用 该 
配置 文件 。 


O 提示 : config 配置 选项 的 缩写 形式 为 -¢<:， 所 以 indexer --config /usr/local/Coreseek/sphinx.conf myindex 与 


indexer -c /usr/local/Coreseek/sphinx.conf myindex 效果 是 一 致 的 。 


(2) --all 生成 所 有 索引 源 

indexer 在 创建 索引 库 时 ， 是 根据 配置 文件 中 的 配置 信息 进行 索引 的 。Coreseek 灵活 及 强 
大 之 处 还 在 于 其 对 分 布 式 的 民 好 文 持 ， 但 在 数据 量 少 的 情况 下 只 需要 配置 一 个 索引 源 即 可 ， 
使 用 all 选项 表示 对 所 有 索引 源 创 建 尝 引 库 ， 该 选项 执行 后 所 有 索引 库 中 的 数据 将 全 部 被 刷 
新 。 使 用 方式 如 下 。 


[root@bogon ~]# cd /usr/local/coreseek/bin/ 
[root@bogon bin] ./indexer --all 
H -4 e DÉI 3 
也 可 以 显示 指定 配置 文件 。 
[root@bogon bin] ./indexer --config /usr/local/coreseek/etc/sphinx.conf --all 


如 果 作 为 计划 任务 执行 时 ， 需 要 完整 的 执行 路 径 ， 并 显 式 指定 配置 文件 。 
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(3) --rotate 索引 轮换 

all 选项 将 索引 配置 文件 中 的 所 有 索引 源 ，rotate 选项 可 以 对 原 有 的 索引 进行 平滑 蕉 换 。 
利用 rotate 选项 ， 可 以 轻易 地 实现 数据 增 量 合并 《〈 即 在 原 有 的 索引 上 增加 新 索引 )。frotate 选 
项 被 执行 后 ， 原 有 的 旧 索 引 将 被 奉 换 ， 这 一 过 程 是 在 服务 不 被 中 断 的 情况 下 进行 鸭 ， 使 用 方 


式 如 下 。 
[root@bogon ~]# cd /usr/local/coreseek/bin/ 
[root@bogon bin] ./indexer --rotate --all 


(4) --merge 索引 合并 

rotate 使 用 狐 的 罕 引 库 蔡 换 旧 索引 库 ， 虽 然 能 够 解决 案 引 更 狐 的 问题 ， 但 在 数据 量 大 的 
情况 下 ， 显 然 会 影 啊 效 率 〈 尽 管 Coreseek 索引 速度 非常 快 )， 理 想 的 办 法 是 将 新 增加 的 索引 
退 加 到 旧 有 索引 中 ， 即 索引 合并 。mersge 选项 能 够 实现 有 尝 引 合并 ， 使 用 方式 如 下 。 


[root@bogon ~]# cd /usr/local/coreseek/bin/ 


[root@bogon bin] ./indexer --merge main xmldata --rotate 

上 述 代码 表示 将 狐 增 加 的 xmldata 索引 库 奶 加 到 旧 MySQL 索引 库 。 通 过 上 述 操作 ， 
Coreseek 将 会 把 xmldata 库 中 的 数据 奶 加 MySQL 索引 库 中 ， 完 成 后 清空 等 待 下 一 次 执行 merge 
操作 。 关 于 多 数据 源 的 配置 将 在 后 面 的 内 容 中 继续 深入 介绍 ， 在 此 该 者 只 需要 理解 即 可 。 


14.3.2 searchd 


searchd 是 一 个 重要 的 命令 工具 ， 无 论 是 前 面 的 配置 文件 ， 或 者 indexer 创建 案 引 ， 这 些 
操作 只 是 进行 全 文 索引 前 的 准备 工作 ， 要 让 Coreseek 能 够 对 外 提供 搜索 服务 ， 后 台 管 理 人 员 
必须 手动 或 使 用 计划 任务 启动 Coreseek 服务 进程 。searchd 工具 就 是 用 于 局 动 Coreseek 服务 
的 ， 下 面 分 别 对 searchd 工具 和 常用 选项 进行 介绍 。 

(1) --help 

help 选项 将 列 出 searchd 工具 的 帮助 选项 ， 人 简写 形式 为 -h。 选 项 运行 后 将 得 到 详细 的 帮 
助 信息 ， 如 以 下 代码 所 示 。 


[root@bogon bin]# ./searchd --help 


Coreseek Fulltext 3.2 [ Sphinx 0.9.9-release (r2117)] 
Copyright (c) 2007-2011, 


Beijing Choice Software Technologies Inc (http://www.coreseek.com) 
Usage: searchd [OPTIONS] 


Options are: 

-h, --help display this help message 

-c, -config <file> read configuration from specified file 
(default is csft.conf) 

--stop send SIGTERM to currently running searchd 


--status get ant print status variables 
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(PID is taken from pid file specified in config file) 
--iostats log per-query io stats 


--cCpustats log per-query cpu stats 


Debugging options are: 


--console run in console mode (do not fork, do not log to files) 
-p, --port <port> listen on given port (overrides config setting) 
-1, --listen <spec> listen on given address, port or path (overrides 


config settings) 


-i, --index <index> only serve one given index 
--nodetach do not detach into background 
Examples: 

searchd --config /usr/local/csft/etc/csft.conf 

(2) --config 


与 indexer 工具 一 样 ，searchd 运行 时 需要 载 入 配置 文件 〈 即 载 入 配置 文件 中 的 searchd 
节点 配置 信息 )， 上 默认 情况 下 载 入 的 配置 文件 为 /usr/local/coreseek/etc/csft.conf 文件 ， 如 果 需 
要 切换 配置 文件 ， 可 以 通过 config 配置 选项 指定 。 使 用 方式 如 下 。 


[root@bogon ~]# cd /usr/local/coreseek/bin/ 


[root@bogon ~]# ./searchd --config /usr/local/coreseek/etc/sphinx.conf 

Copyright (c) 2007-2011, 

Beijing Choice Software Technologies Inc (http://www.coreseek.com) 

using config file '/usr/local/coreseek/etc/sphinx.conf'... 

listening on all interfaces, port=9312 

WARNING: index 'testl': preload: failed to open /usr/local/ coreseek/ var/data/ 

testl.sph: No such file or directory; NOT SERVING 

WARNING: index 'testlstemmed': preload: failed to open /usr/local/coreseek/ var/ 

data/testlstemmed.sph: No such file or directory; NOT SERVING 

WARNING: multiple addresses found for 'localhost', using the first one (ip= 127.0. 

0.1) 

--config 简写 形式 为 -c:， 通 过 该 选项 读者 可 以 指定 任何 合法 的 配置 文件 作为 searchd 运行 
环境 配置 信息 。 

(3) --pidfile 

searchd 主 程 序 局 动 后 ， 系 统 将 生成 searchd.pid 文件 ， 并 保存 到 /usr/local/coreseek/var/log/ 
目录 。 该 文件 是 searchd 主 进 程 与 其 他 进程 进行 通信 的 描述 文件 。 通 过 pidfile 选项 ， 可 以 改 
变 文件 存放 位 置 ， 以 便于 管理 ， 如 以 下 代码 所 示 。 


[root@bogon ~]# cd /usr/local/coreseek/bin/ 


[Iroot@bogon bin] ./searchd --config /usr/local/coreseek/etc/csft.conf --pidfile 


/tmp/searchd.pid 
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(4) --prot 
默认 情况 下 ，searchd 主 进程 启动 后 ， 将 使 用 9312 作为 通信 接口 。 通 过 port 选项 ， 可 以 
修改 款 认 的 通信 接口 。 
[root@bogon bin]# ./searchd --port 9315 
Coreseek Fulltext 3.2 [ Sphinx 0.9.9-release (r2117)] 
Copyright (c) 2007-2011, 
Beijing Choice Software Technologies Inc (http://www.coreseek.com) 
WARNING: --listen and --port are only allowed in --console debug mode; switch 
ignored 
using config file '/usr/local/coreseek/etc/csft.conf'... 
listening on all interfaces, port=9312 
Eim H a PEE ps 命令 检测 9315 Ii H -o 
[root@bogon bin]# ps -aunx |grep 9315 
Warning: bad syntax, perhaps a bogus '-'? See /usr/share/doc/procps-3.2.8/FAQ 
0 7532 0.1 0.8 12004 988 pts/0 S 07:44 0:00 ./searchd --port 9315 
O 7536. 30 O0 5952 75C Ptsyt S+ 07:45 0700 grep 9315 
需要 注意 的 是 ， 更 改 通 信 接 口 后 ， 需 要 在 防火 墙 中 开放 相应 的 通信 接口 ， 否 则 客户 端 
API 将 被 拒绝 链接 。 


14.3.3 search 


search 工具 是 一 个 用 于 命令 行 搜索 测试 的 辅助 工具 。 在 searchd 成 功 局 动 后 ，Coreseek 
束 处 于 激活 状态 ， 此 时 可 以 借助 于 search 工具 进行 简单 的 测试 。 所 有 的 操作 均 是 独立 的 ， 
不 会 影响 Coreseek 数据 检索 效果 ， 也 不 会 对 API 获取 数据 产生 影响 。 简 单 地 说 search 是 一 
个 可 删除 的 工具 ， 但 为 了 方便 测试 ， 建 议 保留 该 工具 。search 工具 第 用 搜索 命令 选项 如 表 
14-1 所 示 。 


表 14-1 search 常用 选项 


使 用 示例 
--config 车 indexer 配置 文件 ./search -c ../etc/csft.conf “ER 
--index i 搜索 测试 节点 ， 即 配置 文件 中 数据 源 节 点 ./search -itestl 苹果 
--limit -l 指定 搜索 返回 的 数据 量 ， 默 认为 20 条 ./search -12 苹果 
Se p n a 2 始 位 置 ， 配 合 limit 选项 可 以 实现 数 jscarch -02 258 


对 搜索 结果 ， 按 照 配置 文件 attr 属性 进行 分 组 ， 类 似 


Ree E ”gsQL PRI GROUP BY 子 语 名 dE 

--sortby -S 使 用 排序 语句 对 结果 进行 排序 ./search -s "@id DESC" 苹果 
--sort=date 搜索 结果 按 日 期 升序 ./search --sort=date ”苹果 
--rsort=date 搜索 结果 按 日 期 降序 ./search --rsort=date ”苹果 
--sort=ts 搜索 结果 按 UNIX 时 间 惟 分 组 ./search --sort=ts R 
--noinfo -q 测试 searchd 及 MySQL 通信 和 是否 正确 ./search --noinfo “ER 
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上 述 选 项 中 ， 除 了 --noinfo 选项 外 ， 其 他 的 搜索 选项 在 API 接口 中 ， 均 有 对 应 的 调用 属 
性 。--noinfo 选项 是 一 个 测试 开关， 运行 结果 如 下 。 

[root@bogon bin]# ./search --noinfo 苹果 

Coreseek Fulltext 3.2 [ Sphinx 0.9.9-release (r2117)] 

Copyright (c) 2007-2011, 

Beijing Choice Software Technologies Inc (http://www.coreseek.com) 

using config file '/usr/local/coreseek/etc/csft.conf'... 

index 'testl': query P ': returned 6 matches of 6 total in 0.005 sec 

displaying matches: 

1. document=1, weight=4 

2. document=4, weight=4 

3. document=5, weight=4 

4. document=2, weight=3 


5. document=3, weight=3 


qd- 


6. document=6, weight=3 

words: 

1. '6 documents, 353 hits 

如 上 述 反 馈 信息 所 示 ，weight 显示 了 MySQL 查询 记录 权重 状况 。 如 果 为 0 表示 处 于 失 
效 状 态 ， 此 时 束 需 要 检测 MySQL dee UC, 


14.4 ”创建 索引 


创建 索引 是 全 文 搜索 引擎 最 关键 的 操作 ， 在 Coreseek 中 可 以 对 多 种 数据 源 创 建 案 引 ， 包 
括 和 常见 的 主流 数据 库 ， 以 及 Xmlpipe2 ~ HTML, TXT 等 。 同 时 ，Coreseek 提供 了 多 种 索引 
关 型 ， 以 满足 各 种 场合 的 全 文 索引 需求 ， 下 和 面 首先 对 索引 源 分 类 进行 介绍 。 


14.4.1 索引 源 分 类 


索引 源 即 为 Coreseek 提供 数据 的 对 象 。Coreseek 文 持 主流 的 数据 文件 、 数 据 库 、 网 页 文 
件 等 作为 数据 源 ， 在 索引 方式 上 都 是 一 致 曲 ， 上 只 需要 在 配置 文件 时 对 索引 源 进 行 配置 即 可 。 
一 个 配置 文件 可 以 同时 配置 多 个 不 同类 型 的 索引 源 ， 也 可 以 将 多 个 索引 源 分 别 定义 不 同 的 节 
点 ， 在 用 户 搜索 时 ， 这 些 索 引 源 将 被 用 户 搜索 。 下 面 将 对 和 音 见 的 索引 源 进 行 介 绍 。 

1. Xmlpipe2 

Sphinx 并 不 是 对 所 有 XML 数据 格式 都 能 够 创建 索引 的 ，Xmlpipe2 是 一 种 由 Sphinx 所 
规范 的 XML 数据 格式 。 开 发 人 员 只 有 遵循 这 些 规范 ，Sphinx 或 Coreseek 才能 识别 索引 | 源 文 
档 。 一 个 简单 的 Xmlpipe2 数据 格式 文档 如 下 代码 所 示 。 


SE EE encocding="urti=8"?> 


<sphinx:docset> 
<sphinx: schema> 
</sphinx:schema> 
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<5phinx: document 16="001"> 


</sphinx:document> 


</sphinx:docset> 

如 上 述 代 码 所 示 ， 该 文档 共 定 义 了 3 对 XML 标签 元 素 。 其 中 <sphinx:docse 亿 元 素 是 
Xmlpipe2 文档 格式 的 根 元 对 ， 只 有 该 元 系 的 存在 ，Sphinx 引擎 才能 解释 ， 人 否则 将 停止 后 续 
索引 操作 ; schema 元 素 是 一 对 可 选 的 Xmlpipe2 元 素 ， 它 用 于 目 定 义 全 文 索引 字段 ， 例 如 标 
题 、 内 容 正文 等 ，document 是 Xmlpipe2 数据 格式 元 素 中 必须 出 现 的 元 素 ， 它 用 于 存放 创建 
全 文 索引 的 schema 上 自 定 义 元 素 值 ， 也 就 是 说 只 有 定义 在 访 元 素 内 的 数据 才 会 被 创建 索引 。 

这 里 需要 重点 对 schema 元 素 进 行 介绍 。 虽 然 schema 元 素 是 可 选 的 ， 但 却 是 非常 重要 
的 。 为 了 便于 理解 ， 这 里 首先 使 用 代码 进行 演示 。 假 设 需要 对 索引 铸 文档 中 的 title 和 content 
字段 创建 全 文 索引 ， 那 么 在 schema 中 定义 的 元 素 如 以 下 代码 所 示 。 


<sphinx:schema> 


<sphinx:rield name=" title" /> 
<sphinx:field name="content"/> 
</sphinx:schema> 
上 述 信息 是 可 以 在 配置 文件 中 定义 的 ， 如 以 下 代码 所 示 。 
# 源 定义 
source xml 


{ 


type = xmlpipe2 

xmlpipe command = cat var/test/test.xml 

# 请 修改 为 实际 使 用 的 绝对 路 径 ， 例 如 : cat /usr/local/coreseek/var/... 
xwlpipe fiele = title 


xmlpipe field = content 

1 

上 述 配置 与 前 面 介 绍 的 Xmlpipe2 schema 子 元 素 配 置 在 作用 上 是 一 致 的 。 正 如 前 面 所 
言 schema 是 可 选 的 ， 但 前 提 是 已 经 在 配置 文件 中 配置 了 相应 的 xmlpipe field 选项 。 也 就 是 
说 ，schema TRF: wll xmlpipe_ field 配置 项 。 

出 于 灵活 性 考虑 ， 直 接 在 schema 元 素 中 定义 xmlpipe field 显然 更 加 友好 (人 至少 不 会 因 
为 更 改 了 不同 数据 源 文 档 ， 叉 要 重新 修改 配置 文件 ， 而 只 需要 在 数据 产 文 档 修改 即 可 )。 所 
以 建议 读者 尽量 在 Xmlpipe2 文档 中 完成 xmlpipe_field 定义 。 

配置 了 xmlpipe_field 索引 字段 ， 还 再 要 实现 xmlpipe_field FE, FMT ELA, F fe 
实现 全 文 索引 。 如 以 下 代码 所 示 。 


<5phinx: document 16="001"> 
title I</ Et LeS 
Content PAA l content: 

</sphinx:document> 


<sphinx:cdocument EE 
<title- P p title 
<content> 新 闻 内 容 2</content> 

</sphinx:document> 


通过 上 述 操 作 ， 结 合 MMSEG 中 文 分 词 插件 ，Coreseek 束 可 以 对 所 定义 的 全 文 文本 字段 
进行 索引 了 。 最 终 Xmlpipe2 文档 如 以 下 代码 所 示 。 


dml yersion="1,. 0" encodinoe="utf- -82?2> 


<sphinx:docset> 
<sphinx: schema> 
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<sphinx:field name="title"/> 
<sphinzs field name= content" /> 
</sphinx:schema> 
<6phinz: document 106="001"> 
<ticle a EE Ee 
<content> 新 闻 内 容 1</content> 
</sphinx:document> 
<sphinx:document id="002"> 
title STI ae 
<content> 新 闻 内 容 2</content> 
</sphinx:document> 
—/epbinsgrdocsetz 


通过 前 面 的 介绍 ， 相 信 读 者 已 经 对 Xmlpipe2 数据 格式 有 了 一 些 认识 。 事 实 上 Xmlpipe2 
主要 的 标签 元 系 只 bn 5 个 ， 其 他 都 是 可 目 定义 的 。 如 表 14-2 所 示 。 


表 14-2 Xmlpipe2 文档 格式 元 素 


Xmlpipe2 元 素 


sphinx:docset Xmlpipe2 文档 根 元 素 JE 

sphinx:schema 文档 模式 ， 如 果 定 义 ， 必 须 位 于 第 1 个 元 素 个 

sphinx:field schema 子 元 素 ， 用 于 定义 全 文 索引 字段 Gr 

l sphinx:schema 子 元 素 ， 作 用 类 似 于 sphinx:field， 但 提供 了 字段 数据 类 型 Ss 

sphinx:attr T 
限制 

sphinx:document sphinx:docset 子 元 素 ， 存 放 全 文 索引 的 文本 字体 是 


默认 情况 下 ， 使 用 sphinx:field 元 又 定 义 的 字段 为 文本 字段 。 使 用 sphinx:attr "CSS Ve 
段 时 ， 可 以 对 字段 的 数据 类 型 进行 限制 。sphinx:attr 文 持 的 数据 类 型 分 别 如 下 : 

> int: 数值 整 型 。 

> timestamp: UNIX JEJER- 

> str2ordinal: 字符 串 关 型 。 

> bool: 布尔 值 。 

> float: 浮 点 数值 类 型 。 

> multi: pg 

这 些 数 值 类 型 通过 sphinx:attr 元 素 中 的 type 属性 指定 。 假 设 定义 一 个 名 为 author id， 类 
型 为 int 的 字段 ， 代 但 如 下 。 


<sphinx:schema> 


<sphinx:attr name="author 1d" type= "Int default="]"/> 


</sphinx:schema> 

sphinx:attr 元 系 还 文 持 以 下 属性 列表 。 

> name: 字段 名 称 。 

> bits: 默认 情况 下 ，int 数据 类 型 长 度 为 32 位 ， 可 以 通过 bits 属性 项 修改 int 数值 

长 度 。 

> default: 为 字段 提供 一 个 默认 值 。 

直 要 注意 的 是 ， 无 论 字 段 定 义 为 何 种 数据 类 型 ， 字 段 的 数据 量 最 大 不 能 超过 2MB, m 
则 Coreseek 将 对 字段 中 的 数据 进行 截取 。 这 藉 意 味 大 Xmlpipe2 文档 并 不 适合 海量 型 的 数 
据 ， 如 末 有 此 需要 ， 可 以 使 用 数据 库 索 引 源 。 
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2. 数据 库 索 引 源 

Coreseek 对 第 见 的 开源 数据 提供 了 很 好 的 文 持 ， 包 括 MySQL 和 PostgreSQL 。 在 
Windows 环境 下 还 实现 了 对 Microsoft SQL Server 数据 库 原 生 文 持 。 通 过 内 置 开 放 性 接口 ， 
例如 ODBC 微软 开放 数据 库 互 联 )、Python 数据 源 等 ，Coreseek 能 够 轻松 地 实现 万 能 数据 
Rm, ml krut, 


source mysql 


{ 


type = mysql 

SEL NOST = localhost 

sql user = TOOT 

sql pass = FOOL 

sal elo = tp 

SaL Bort = 3306 

sql query pre = GEI NAMES UTFG 


s@qL query = SELECT ic, UNIX TIMESTAMP (ace) time) AS times, CLtle, Content PROM 
tok article 
SQL query 1nfe = OELECT A FROM pk articole WHERE Id 1q 


} 

如 上 述 代 码 所 示 ， 配 置 项 type 定义 了 数据 库 连 接 类 型 。 在 Linux FS FRÆ A iK 
外 安装 驱动 ) 支持 的 驱动 有 MySQL、psSQL; 在 Windows 平台 下 ， 原 生 支 持 的 驱动 有 
MSsql、MySQL、psSQL、ODBC。 开 发 人 员 可 根据 需要 自行 选择 。 

对 于 Coreseek 而 言 ， 数 据 库 连接 是 由 中 间 件 来 完成 后 。 对 于 开 有 人员 而 言 ， 中 间 件 提供 
了 一 套 接 口 ， 通 过 配置 接口 选项 数据 ，Coreseek 惑 能 够 完成 不 同类 型 数据 库 的 连接 及 切换 。 
最 终 只 需要 修改 配置 项 type 值 即 可 。 常 见 的 数据 库 配 置 项 如 表 14-3 所 示 。 


表 14-3 数据库 配置 项 


连接 数据 库 配 置 〈 仪 对 原生 文 持 的 驱动 生效 ) 


sql host 数据 库 连 接地 址 空 〈 必 选 ) 
sql user 数据 库 认 证 用 户 名 室 〈 必 选 ) 
Sql Doss 数据 库 认 证 用 户 密码 室 〈 必 选 ) 
sql db 选择 数据 库 名 称 T OU) 
sql_port 数据 库 通信 端口 3306 (mysql), 5436 (pssql) 
sql sock 连接 到 本 地 SQL 服务 器 时 使 用 的 UNIX socket 名 称 ZG Col 
数据 查询 配置 
sql_query_pre 预 查询 ， 即 设置 查询 前 的 环境 《例如 查询 编码 ， 查 询 缓存 等 ) ” 空 ( 可 选 ) 
sql_query 主 碍 询 ， 结 果 将 作为 索引 数据 来 源 室 〈 必 选 ) 
sql query range 分 区 查询 空 〈 可 选 ) 
sql range step 区 块 查 询 步 进 1024 
sql_attr_uint 声明 无 符号 整数 属性 空 〈 可 选 ) 
sql_attr_timestamp Linux 时 间 惟 排序 SS 
sql query info 命令 终端 显示 查询 语句 SS 
sql query Post 后 查询 ， 与 sql _ query _ pre 作用 相反 SN 
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配置 好 数据 源 后 ， 旋 者 可 以 使 用 表面 介绍 的 indexer、searched 工具 测试 是 否 配置 正 篆 。 
如 果 一 切 正 常 束 可 以 使 用 Coreseek 高 效 索 引 引 警 创 建 全 文 索引 数据 了 。 


14.4.2 ” 增 量 索引 


剖面 介绍 了 使 用 indexer 创建 索引 的 过 程 。 在 多 数 情 况 下 ， 因 为 Coreseek 85ER ot 
10MB/S$， 上 所 以 只 需要 创建 一 个 索引 源 即 可 满足 需求 。 但 是 在 数据 量 随 时 激增 的 大 型 应 用 中 
(如 SNS、 评 论 系统 等 )， 单 一 的 索引 源 将 会 给 indexer 造成 极 大 的 性 能 负 衙 。 

增 量 索 引 能 够 在 一 定 程度 上 提升 Coreseek 索引 性 能 ， 降 低 CPU 使 用 率 。 增 量 索 引 的 原 
理 非 党 简单 ， 即 使 用 “ 主 索引 + 增 量 索 引 ” 的 方式 创建 案 引 。 其 中 主 索 引 表 存放 50% 以 上 的 
数据 量 ， 增 量 索 引 表 通 币 只 需要 存放 较 近 插入 的 数据 即 可 《〈 增 量 索 引 表 人 允许 由 多 个 索引 表 组 
成 )。 增 量 索 引 是 通过 主 从 索引 继承 实现 的 ， 继 承 的 格式 为 “ 增 量 索 引 : 父 索引 ” 如 以 下 代 
公所 未 。 

| 


source mam 


í 


} 

Stu 

index main 

l 

} 

#main 的 增 量 索引 表 
source topicl:main 
{ 

} 

index topicl:main 


{ 


} 
# 配 置 索 引进 程 参数 


indexer 

{ 

} 

# 配 置 搜 索 进程 参数 
searchd 


{ 


MIAR, Je OI un TERIERA. KAEH DL ir AER 
d 


WéI, IERT Dette ole em, Jl kr, 
# 
i Minimal Sphinx cContiguration samole (elean, simple, functional) 
# 
source main 
i 
type = mysql 
sdg Mino t a 
sql user = TOOT 
saql pase = EOOLE 
sal dlio = tp 
sapore = 35306 i Optional, cefault is 3306 
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saql uery pre = SET EE 

sql Guery pre = SET NAMES UTEG 

sql query = SELECT id,title,content FROM tpk_article where id<10 
sql attr uint = ade) time 

sql attr cimestamp = 1e 

sa auer EC 


} 


1noex maLm 


source = main 

path = /usr/local/coreseek/var/data/main 
docinto = extern 

#charset type = Sbcs 

charset dictpath = /usr/local/mmseg3/etc/ #Linux 环境 需要 声明 

charset _ type = zh cn.utf-8 # 中 文 分 词 必须 使 用 zh cn.utf-8 编码 


} 


source artLlc]lel -maln 


{ 
sql Guery pre = SET NAMES UTEG 
sql _ query = SELECT id,title,content FROM tpk_article where id>10 


Lacex arciclel s: main 


{ 


source = articlel 
和 
charset dictpath = /usr/local/mmseg3/etc/ #Linux 环境 需要 声明 
charset type = zh cn.utf-8 # 中 文 分 词 必 须 使 用 zh cn.utf-8 编码 
} 
indexer 
í 
mem limit = 32M 
} 
searchd 
f 
port = 9312 
log = /usr/local/coreseek/var/log/searchd.log 
query log = /usr/local/coreseek/var/log/cowery. Log 
reac Timeout = 5 
max Chileren = 30 
pid file = /usr/local/coreseek/var/log/searchd.pid 
max marches = 1000 
seamless rotace = 1 
preopen indexes = 0 
unlink olal = 


} 

上 上述 配 置 文件 定义 了 一 个 main 主 索 引 和 一 个 articlel 增 量 索引 。 为 了 方便 演示 ， 
里 将 主 索引 数据 范围 限制 为 id<10 的 数据 ;而 在 articlel 增 量 索 引 中 限制 id>10 see 
这 样 束 实现 了 主 从 索引 〔( 即 增 量 索 引 )。 在 合并 主 从 索引 前 ， 首 先 需 要 生成 增 量 索引 ， 合 
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令 如 下 。 


[root@~]cd /usr/local/coreseek/bin 

[root@bogon bin] ./indexer article) 

增 量 索引 生成 后 ， 只 需要 使 用 前 面 介绍 的 merge 选项 将 增 量 索引 表 合 并 到 主 有 索引 表 即 
可 。 完 成 索引 合并 后 ， 可 以 使 用 search 工具 测试 搜索 效果 。 


[root@bogon binlcd /usr/local/coreseek/bin 


[root@bogon bin] ./search facebook 


Ion, BR mK RERI ASHI ER S Aj EPR. 


[root@~]cd /usr/local/coreseek/bin 


[root@bogon bin] ./indexer --merge main articlel --rotate 


14.4.3 ”实时 索引 


Coreseek 是 一 个 独立 的 索引 服务 器 ， 所 有 新 增 的 数据 必须 通知 Coreseek 才 会 创建 新 增 的 
数据 索引 。 也 就 是 说 ，Coreseek 并 没有 实时 索引 的 功能 ， 开 发 人 员 需 要 通过 第 三 方 监 探 脚 本 
或 接口 实现 实时 索引 。 当 前 最 常用 的 方法 有 两 种 : 一 种 是 使 用 SphinxSE 插件 ， 另 外 一 种 是 
使 用 数据 库 索 引 统计 计数 项 。SphinxSE 是 一 个 基于 Sphinx 索引 引擎 的 MySQL ENRI IYA 
件 ， 在 使 用 方式 上 与 MySQL 全 文 检索 相 类 似 ， 但 由 于 只 能 对 MySQL 生效 ， 并 且 需 要 特定 
的 MySQL 版 本 ， 所 以 这 里 不 建议 人 使用。 下面 将 使 用 统计 计数 需 实 现实 时 索引 。 

1. 实现 思路 

顾名思义 ， 实 时 索引 就 是 狐 增 数据 后 ， 索 引 服 务 占 束 能 够 并 即 (或 在 较 少 时 间 内 〉 对 新 
增 的 数据 创建 全 文 罕 引 。 利 用 前 和 面 介绍 的 增 量 索 引 ， 再 配合 Linux 计划 任务 机 制 ， 可 以 实现 
狭义 上 的 实时 索引 。 但 是 这 种 方式 只 适用 于 数据 增长 较 慢 的 场合 。 如 果 数 据 增 长 较 快 ， 增 
量 索 引 本 喘 就 是 一 个 海量 型 的 数据 库 ， 较 短 时 间 内 对 增 量 索引 进行 索引 及 合并 操作 无 疑 是 不 
合理 的 。 

实时 索引 的 基本 腺 理 束 是 以 增 量 索 引 为 基础 ， 然 后 在 当前 索引 源 数 据 库 中 创建 一 个 统 
计 表 ， 用 于 统计 当前 索引 数据 表 最 大 id 什 ， 每 次 创建 增 量 索引 时 ， 在 伍 询 〈sql_query 
post) 中 使 用 SQL 获取 数据 库 表 记录 最 大 id 值 ， 并 日 将 该 值 更 狐 到 统计 表 。 利 用 最 大 id 
值 可 以 实现 每 次 创建 增 量 索引 时 均 从 该 id 值 开始 。 同 时 在 创建 主 索 引 时 ， 将 主 索 引 库 的 索 
引 范 围 限于 该 id 值 内 〈( 增 量 索 引 应 大 于 统计 id 值 )， 并 使 用 预 查询 (sql query pre) 更 新 
统计 表 id 值 。 

通过 上 述 步 召 ， 每 次 执行 增 量 索 引 束 相当 于 对 狐 增 数据 创建 全 文案 引 。 结 合 Linux 计划 
任务 机 制 ， 束 可 以 完美 地 实现 实时 索引 了 。 下 面 将 结合 示例 讲解 实现 过 程 。 

2. 实现 过 程 

实时 索引 的 核心 是 计数 器 ， 所 以 首先 需要 在 数据 库 中 创建 一 个 统计 表 。 这 里 创建 的 统计 
表 名 为 sph counter， 该 表 共 有 两 个 int 类 型 字段 ， 如 以 下 SQL 代码 所 示 。 


CREATE TABLE Spa Counter 

( 
Counter 16 INTEGER PRIMARY KEY NOT NULG, 
max doc id INTEGER NOT NULL 
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代 公 运行 后 ， 表 结构 如 图 14-10 所 示 。 


-T> counter id max doc id 
PE 1 48 


图 14-10 sph counter 统计 表 结 构 


统计 表 创 建 完 成 后 ， 只 需要 重新 配置 /etc/csft.conf 文件 即 可 ， 如 以 下 代码 所 示 。 
# 

i Minimal Spain: Configuracion samole (clean, simole, functional) 

# 

source main 


{ 


type = mysql 

SE ek SE 

sql user = Ceiba 

SG pases = 535416 

sal dla = tp 

So Onl = 53506 六 Optional, default is 3306 
saql Guery pre = SET SESSION query EE 

sal cuery pre = OEL NAMES UTEG 


sql query pre = REPLACE INTO sph counter select 1,max(id) from tok article 


sal Guerry = SELECT id, title,content FROM tpk article \ 
where 16 <= (select mex doe ic from sph COunter where Counter 
id=1) 
sql attr uing = 人 CQ time 
sql attr timestamp = id 
5G) GUery rnio =SEEEOCD TERM ED arei ele Mme gd ro 


b 


index main 


| 


source = main 
parth = /usr/local/coreseek/var/data/main 
cocinfe = extern 


charset dictpath = /usr/local/mmseg3/etc/ 
charset type = zn Cmn utEE=8 
) 
source articlel:main 
{ 
sql query pre = SET NAMES UTEG 
sql guery = SELECT id,title eontent FROM tpk article \ 
wnere 16 >(select max Coe id from sph cownter where Counter 16s1) 
sql query post = REPLACE INTO span Counter select 1, max(icd) rrom tok article 


index articlel:main 
í 
source = articlel 
Ben E T er e ea a a a e 
charset dictpath = /usr/local/mmseg3/etc/ 
Charset type = zn cn.utf-=8 
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} 


indexer 
| 
mem Limit = 32M 
} 
searchd 
{ 
port = 9312 
log = /usr/local/coreseek/var/log/searchd.log 
query log = /usr/local/coreseek/var/log/query.log 
reall timeout = 5 
max Chileren = 30 
pid file = /usr/local/coreseek/var/log/searche -piec 
max matches SOOO 
seamless rotate = 1 
preopen 1ndexeg = 0 
unlink olal = l 


} 

wW ERWE m 0 ole ok ré OT articlel, sph counter 表 统 计 记 录 将 发 生 
相应 变化 。 整 个 过 程 与 前 面 介 绍 的 实现 思路 是 一 致 的 ， 接 下 来 将 通过 测试 ， 观 察 数 据 变 化 。 

3. 测试 实时 索引 

为 了 更 好 地 进行 测试 ， 首 先 将 原 有 的 main 及 articlel St Hinz, 

[root@bogon bin]# rm -rf /usr/local/coreseek/var/data/* 

然后 创建 主 索引 〔 即 main 索引 )。 


[root@bogon /]# cd /usr/local/coreseek/bin/ 


[root@bogon bin]# ./indexer main 

主 索 引 | 创建 完成 后 ， 此 时 观察 sph counter 表 ， 可 以 看 到 max doc id 字段 值 将 更 新 为 
tpk_article 表 id 字段 最 大 值 ， 假 设 为 10。 接 下 来 继续 创建 增 量 索引 ， 为 了 方便 演示 ， 肯 先 在 
tpk_article 数据 表 中 插入 两 条 数据 ， 这 两 条 数据 将 作为 增 量 索引 ， 同 时 id 最 大 值 为 12 (max_ 
doc id 值 依旧 为 10)。 


[root@bogon bin]# ./indexer articlel --rotate 


上 述 命 令 执 行 后 ，max_doc id 将 会 更 独 为 12， 同 时 增 量 索引 数据 库 中 将 新 增 两 条 数 


据 ， 证 明 配 置 正确 无 误 ， 如 果 数 值 没有 更 新 ， 需 要 仔细 检测 sql query post HN rbl dt 
确 。 读 者 可 以 使 用 前 面 介绍 的 search 工具 ， 测 试 系 引 效 果 。 


4. 创建 监控 脚本 

接 下 来 将 创建 一 个 Shel 监控 脚本 ， 用 于 实时 目 动 创建 增 量 索引 |。 
[root@bogon src]#mkdir -p /home/src 

[root@bogon srcl#cd /home/src 


[root@bogon srcl# touch articlel.sh 


[root@bogon src]# chmod +x articlel.sh 


articlel.sh 脚本 文件 代码 如 下 所 示 。 
#!/bin/sh 
Zcoreseek, sh 


/usr/local/coresesk/bin/indezsr articlel --rotate 


最 后 只 需要 将 该 脚本 加 入 corntab 计划 任务 即 可 。 
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[E00ECBOgOn Sre]# crontab -ë 

SJ rk = + /home/src/articlel:sh >> /tmp/articlel crontab.log 

ERRER AEN 2min 执行 一 次 articlel.sh 脚本 ， 最 终 达 到 模拟 实时 索引 的 效果 。 访 者 
可 以 在 tpk article 数据 表 中 插入 数据 ， 并 观察 sph counter 统计 表 变 化 情况 。 


14.5 在 MVC 中 搜索 数据 


Coreseek 索引 服务 咒 局 动 后 最 终 还 需要 借助 客户 端 为 用 户 提供 搜索 服务 。 Sphinx 官方 提 
供 了 常见 开发 语言 API， 这 些 API 同样 适用 于 Coreseek。 对 于 PH 开 友 而 言 ， 官 方 提供 了 
sphinxapi.php 接口 文件 ， 该 接口 能 够 完成 对 Sphinx 搜索 服务 的 常见 操作 。 同 时 ，Sphinx È 
方 还 提供 了 PHP 扩展 模块 ， 通 过 扩展 模块 同样 能 够 方便 地 完成 Sphinx 数据 搜索 ， 而 且 在 效 
率 及 速度 上 上 共有 明显 的 优势 。 下 和 面 首先 介绍 PHP 扩展 模块 的 安装 。 


14.5.1 安装 Sphinx 扩展 模块 


Sphinx 扩展 模块 是 PHP 官方 推荐 的 扩展 之 一 ， 在 安 北方 式 上 与 前 面 革 节 介绍 的 
Memcache, Redis 等 扩展 模块 安装 方式 类 似 。 不 同 之 处 在 于 安装 Sphinx 扩展 模块 前 ， 首 先 需 
要 安装 libsphinxclient 依赖 包 。 下 向 分 别 介绍 。 

(1) 安装 libsphinxclient 依赖 包 

libsphinxclient 依赖 包 由 Sphinx 官方 提供 ， 读 者 可 以 在 Coreseek 3.2.14 源 个 包 中 找到 该 
JÆ (EH Coreseek-3.2.14/csft-3.2.14/apilibsphinxclient/)。 将 该 类 库 复 制 到 需要 安装 Sphinx H 
展 模 块 的 服务 器 ， 然 后 执行 编译 安装 命令 即 可 ， 过 程 如 下 。 


[root@~]# cd /datal/coreseek-3.2.14/csft-3.2.14/api/libsphinxclient/ 


[root@bogon libsphinxclient]#./configure 


[root@bogon libsphinxclient]# make 


[root@bogon libsphinxclient]# make install 


通过 上 述 操作 ，libsphinxclient 依赖 包 束 安装 完成 了 ， 接 下 来 将 进入 Sphinx 扩展 模块 的 


(2) 安装 Sphinx 扩展 

与 其 他 PHP 扩展 模块 一 样 ， 安 装 Sphinx 扩展 主要 分 为 两 个 步骤 。 首 先 使 用 phpize T 
具 编 译 安装 Sphinx 扩展 模块 ， 然 后 在 php.ini 配置 文件 中 添加 相应 的 扩展 模块 即 可 。 操 作 
步骤 如 下 。 
[root@~]# cd /datal 
[root@bogon datal]# tar zxvf sphinx=-1.-1.0.tgz 
[root@bogon datal]# cd sphinx-1.1.0 


[rootEbogon sphinx-1.1.0]?F /usr/local/php/bin/phpize 


[root@bogon sphinx-1.1.0]#./configure --with-php-config=/usr/local/php/bin/php- 
config --with-sphinx 


[root@bogon sphinx-1.1.0]# make && make install 
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Build complete. 
Don't forget to run 'make test'. 
Installing shared extensions: /usr/local/php/lib/php/extensions/no-debug-non- 


25-200 906267 


过 上 述 操作 ，Sphinx 扩展 模块 束 安 装 完 成 了 。 可 以 通过 查看 PHP 扩展 目录 ， 检 查 
e 展 模块 文件 是 否 存 在 。 


[root@bogon sphinx-1.1.0]# ls /usr/local/php/lib/php/extensions/no-debug-non-zts- 
20090626/ 

sphinx. so 

返回 结果 显示 sphinx.so 文件 成 功 生 成 ， 表 明 Sphinx 扩展 模块 安装 成 功 。 

(3) 配置 Sphinx 扩展 模块 

接 下 来 上 只 需要 在 php.ini 文件 中 加 入 sphinx.so 模块 声明 即 可 。 


[root@bogon sphinx-1.1.0]# vi /usr/local/php/etc/php.ini 


;extension=php shmop.dll 


extension=sphinx.so 


最 后 只 需要 重启 php-fpm， 以 便 痢 配置 文件 生效 。 


[root@~]# pkill php-fmp 


[root@~]# /usr/local/php/sbin/php-fpm 
重新 局 动 后 ， 通 过 phpinfo 检测 PHP 运行 环境 ， 如 果 存 在 Sphinx 选项 ， 证 明 Sphinx 扩 
展 模 块 运行 正常 。 如 图 14-11 所 示 。 


图 14-11 Sphinx 扩展 模块 


PJE, Sphinx 扩展 模块 就 安 关 完 成 了 。 开 发 人 员 可 以 开发 基于 Coreseek 或 Sphinx 全 文 
搜索 服务 器 了 了。 


14.5.2 ”使 用 PHP 接口 


事实 上 Sphinx 客户 端 与 服务 端的 通信 模式 使 用 的 驶 是 Socket 通信 ， 所 以 只 要 掌握 
Socket 通信 ， 使 用 任意 的 编程 语言 均 可 以 实现 Sphinx 搜索 服务 。 同 样 ，Sphinx 官方 提供 的 
PHP 次 库 接口 也 是 基于 Socket 实现 的 ， 所 以 在 使 用 该 类 库 前 ， 首 先 需 要 确保 当前 PHP 服务 
器 已 经 开局 Socket 通信 文 持 。 通 过 phpinfo 可 以 检测 到 相应 选项 ， 如 图 14-12 所 示 。 
Sockets Support [enabled 


图 14-12 Ez) Socket 通信 支持 


如 图 14-12 所 示 ， 表 示 当 前 PHP 环境 支持 Socket 通信 。 一 切 就 绪 后 ， 将 csft-3.2.14/ 
api/sphinxapi.php 文件 复制 到 本 地 网 站 即 可 。 接 下 来 址 可 以 在 PHP 中 直接 调用 了 ， 如 以 下 代 
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Wir a. 


<?php 
require ("sphinxapi.php"); 
Şsp=new SphinxClient(); 
Sconn=$sp=>s6t Server ("192,168,2 16"; 9312); 
rass  qguery (urldecode (T Ni) nAn, 
var dump (> rS); 

2 


如 果 能 够 正确 返回 数组 信息 ， 表 明 sphinxapi.php 运行 正常 。 


O 提示 : 无 论 是 使 用 sphinxapi.php 类 库 还 是 使 用 Sphinx 扩展 模块 作为 API， 在 客户 端 连接 时 ， 必 须 允 许 
Sphinx 服务 通信 端口 接受 外 部 访问 (默认 通信 端口 为 9312) ， 建 议 将 Sphinx 搜索 服务 器 放置 于 内 网 并 关 
闭 所 有 防火 墙 ， 以 提高 服务 器 啊 应 能 力 。 此 外 ，PHP 5.2.17 版 本 Socket 功能 存在 严重 缺陷 ， 使 用 该 版 本 
访问 Sphinx 服务 器 时 ， 将 得 不 到 搜索 结果 。 


14.5.3 ”在 MVC 中 搜索 数据 


连接 Coreseek 索引 服务 器 后 ， 束 可 以 调用 API 提供 的 成 员 方 法 进行 全 文 搜索 了 。 接 下 
来 将 继续 使 用 ThinkPHP 搜索 Coreseek 服务 器 中 的 数据 ， 帮 助 读者 加 深 对 全 文案 引 的 认识 。 
这 里 将 使 用 Search 控制 费 作 为 搜索 栏目 。 下 和 面 首先 创建 搜索 入 口 。 


1. 搜索 入 口 
搜索 入 口 是 用 户 进行 搜索 前 的 一 个 UI 界面 ， 这 里 将 使 用 index 动作 作为 搜索 入 口 ， 如 
以 下 代码 所 示 。 
<?php 
class SearchAction extends Action{ 
/** 


GER 

x Enter description here ... 

SE 

publie function index (|) i 
C ("LAYOUT ON", false) 5 
EE EE 
Erres 


} 


在 上 述 代码 中 ， 首 先 屏蔽 ThinkPHP 布局 功能 ， 然 后 泻 染 index.html 模板 即 可 ， 代 码 如 
TIT: 


ere 
<O1vy ClasSs=" Sarcn block"> 
SE ELSE EE Ee EE ce oUF Se rem e Eiere Se 
< 
en a 
人 
</span></li> 
</form> 
SE 


ee 
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SE EE Se 
SS E 
EE 


由 于 篇 幅 所 限 ， 这 里 并 没有 列 出 index.html 模板 的 所 有 代码 ， 读 者 可 以 在 本 书 附送 的 源 
代码 包 的 home/Tpl/Index/ 目 录 中 找到 该 文件 的 完整 代码 。 界 面 效 果 如 图 14-13 所 示 。 


"Firefox OO EN x 


é > Q G | @ tp.localhost/index.php/search/index/search Ce ee | wë: AARE EI 会 


我 的 博客 开通 了 ， 欢 迎 来 访问 我 的 博 ! 
我 将 在 这 里 对 总 结 我 的 学 习 成 果 ,不定 


各 位 共同 探讨 ， 共 同 进步 。 


网 站 留言 
暂时 没有 留言 


图 14-13 搜索 入 口 界面 


单 击 “ 全 文 搜索 ”按钮 后 ， 首 先 触 发 search 脚本 函数 ， 在 该 函数 中 对 用 户 输 入 的 信息 进 
行 检验 。 脚 本 代码 如 下 所 示 。 


<sCript type=" text/javascript" EE EE Sofc.nect/ /juery.J8" > 
</script> 


SEET type=" text/javascript" > 

function search (){ 

Li ($ ("words") oval () , lengtn<1) 4 
Sin. search result bar) himi ("SEA T); 
return false; 


} 


ELCLUEN CEUES 


| 
</3Cript> 


MERE, KEKSE AAE hS FH search 动作 。 在 该 动作 中 需要 调用 Sphinx 
API 实现 全 文 检 索 。 

2. 全 文 搜索 

用 户 提 交 关 键 词 后 ， 将 被 提交 到 search 动作 进行 处 理 。 为 了 方便 讲解 ， 接 下 来 将 使 用 
Sphinxapi.php 类 库 进 行 搜索 操作 ， 在 实例 化 Shpinx 类 库 之 前 ， 首 先 需 要 在 ThinkPHP 中 导入 
相应 的 类 库 文 件 。 完 成 上 述 操作 后 ， 接 下 来 就 可 以 使 用 SphinxClient 对 象 成 员 方法 进行 全 文 
搜索 了 。 如 以 下 代码 所 示 。 

SE 


» ENnEer EE TEE ENEE oso 
ER 
SEI 

C ("LAYOUT ON", false); 


// 导 入 系统 扩展 函数 库 
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RE OR Punction extend", E EE EE er 
EE 

EE (OR 

$sp=new SphinxClient (); 

Sconn=$sp=>s6etServyer ("192,168,216 9312) 5 


IE S ery a ar eco DEEG 
die ( "Errors ™® . $sp=>getLassSalL () ) 5 
exit ( 0m); 

} 

EE 

Set = mull; 

/ /提取 key 

foreach ( Sstrs as Skey => Svalue ) { 

Sse rae 

| 

// 得 到 所 有 文章 id 

particle id- Erim St 

// 查 询 MySQL 数据 库 


Sarticle=M(vArEticle")s 

Eer er eer lee Et 

Seel eebe Deen Eet ek 
WEE 

Şrows=Ş$article->where (Ş$data)->select(); 

Stnis ->assicon("list", $rows)s 


Stnhnis=>display() s 
} 
search 动作 对 应 的 search bm) 视图 模板 代码 如 下 所 示 。 


<include file="search head" /> 
<div id="search"> 
SE Class=" search lList™> 
<input type="hidden" id="search Info" value "< = [Searchn info- "T" 7> 
<foreach name="list" item="vo"> 
<li><a href="; W><]l]=--= y0, Title} ==></a> 
<p><!--{Ş$vo.content|msubstr=###, 0, 190}--></p> 
J 
</foreach> 
</div> 
SE EE 
<include Cilescitsearch foot" /> 


全 此 ， 一 个 简单 的 全 文 搜索 引擎 融 完 成 了 ， 效 末 如 图 14-14 所 示 。 

3. 数据 分 页 

AAU P Sphinx 每 次 搜索 最 多 返回 10 条 记录 ， 开 发 人 员 可 以 通过 setLimits 方法 设 
转 仿 移 量 及 返回 记录 数量 ， 实 现 数 据 分 页 功能 。 如 以 下 代码 所 示 。 

Aa 


* 搜索 返回 数据 


x Enter description here ... 
SCH 
public function search|) i 
C ("LAYOUT ON", false); 
// 导 入 系统 扩展 函数 库 
LIMOT ("ExtCenc. Funct1ion .extenc” , “Tn1inkeABP", paa) 
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欢 吕 光合 我 的 博客 


首页 | 所 有 日 志 | 搜索 中 心 | 发 表 日 志 


ES 


库 克 就 苹果 地 图 缺陷 向 用 户 道歉 ` 未 提供 一 流 


网 易 科技 讯 9 月 28 日 晚间 消息 ， 苹 果 CEO 蒂 姆 . 库 克 今日 就 苹果 地 图 存在 诸多 功能 问题 向 用 户 发 函 致 舱 ， 称 未 能 为 用 户 提供 世界 一 流 的 
产品 体验 感到 抱 款 。 这 是 苹果 首次 回应 近来 备 受 斋 病 的 iOS 6 地 图 服务 。 4 上 周 ， 苹 果 对 此 事 唯 一 给 出 的 评论 就 是 ，“ 我 们 还 处 于 起 步 阶 
Er. IST, DES SCDE CHE ; 但 是 ' 随 着 时 间 的 推移 , 苹果 想 为 客户 提供 更 好 的 地 图 功能 -; 包括-… 
夏普 称 会 制造 足够 iPhone5 所 需 的 显示 屏 


网 易 科技 讯 9 月 28 日 消息 ， 据 国外 媒体 报道 ， 夏 普 已 经 制造 了 足够 用 于 苹果 iPhone5 的 显示 屏 。 这 也 表明 ， 屏幕 供应 的 瓶颈 可 能 已 经 得 
IER. 分 析 师 曾 指 麦 ， 屏 幕 供应 的 短缺 导致 本 月 新 推出 的 iPhone5 无 法 满足 市 场 不 断 增 长 的 需求 。 据 消息 人 士 称 ，8 月 底 ， 即 
苹果 流 媒 体 电台 服务 负 障 碍 : 素 尼 /ATV 对 价格 不 满 


网 易 科技 讯 9 月 29 日 消息 ， 据 国外 媒体 报道 ,苹果 的 互联 网 电台 服务 计划 最 近 在 获得 由 索尼 /ATV 控 制 的 音乐 的 授权 方面 时 到 困难 ， 索 尼 
/ATV 是 此 前 由 迈克 尔 -杰克 还 (Michael Jackson) 和 索尼 合资 成 立 的 企业 ， 最 近 成 为 了 全 球 最 大 音乐 发 行商 。 苹果 与 索尼 /ATV 的 争议 在 
于 ,苹果 在 其 服务 上 流 媒 体 描 放 索 尼 人 TV 的 了 歌曲 应 该 支付 的 价格 ， 但 目前 的 情况 是 ， 苹 有 果 需 要 与 全 部 音乐 公司 进行 … 


图 14-14 全文 搜索 效果 


EE ee Er ords Ie 
ER 
$sp=new SphinxClient () ; 
Sconn=$sp=>s96etServyer 人 192. ,9312); 


SE EE Eeler eet 

SEET 
die EE EE DEER 
exit ( 0 ); 


} 
On E 


SPade = new Page (Şras["total"],4)7 
Sshow = $Page->show();// 分 页 显示 输出 


$sp->setLimits ($Page->firstRow,$Page->listRows) :; 
Srs = $sp=>cuery ( $words, =W )s 


ss a aee |e 

vst - null; 

/ / KER key 

foreach ( S$strs as Skey => Svalue ) { 
SST = Svalmse, 

i 

MA VC 74 

SEET 

// 查 询 MySQL 数据 库 


EH EE EAR 
Ser ee Eeler 
Şrows=$article->where ($data)->select(); 
$search info=" RFX Ee 
(nt äimen! .uäkbn- 
EE ERKESKR 
SE Ee Bette eebe 
Sthnis ->assion("list", Sows)s 
} 
sthis >displav(): 


} 
如 上 述 代码 所 示 ， 在 实现 分 页 的 原理 上 与 传统 的 MySQL 数据 分 页 相 类 似 ， 不 同 的 是 由 
SQL 语句 变 成 setLimits 方法 。 任 何 的 记录 只 要 满足 网 个 条 件 即 可 实现 数据 分 页 〈 即 总 记录 
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JAIN", 这 里 使 用 $ras["total"] 获 取 总 记录 ， 仿 移 量 则 由 Page 类 计算 。 最 后 只 需要 在 页 面 
适当 位 置 放置 $page 变量 即 可 。 侍 此 ， 一 个 带 有 分 页 功能 的 全 文 搜 索引 擎 就 完成 了 人， 效果 如 
图 14-15 所 示 。 


欢迎 光 | 临 我 的 博客 


首页 | 所 有 日 志 | 搜索 中 心 | 发 表 日 志 


SR 
关键 字 为 【苹果 】 共 有 11 条 记录 ,用 时 0.148 秒 


库 史 就 苹果 地 图 缺陷 向 用 户 道歉 : 未 提供 一 流体 验 


网 易 科 技 讯 9 月 28 日 晚间 消息 ， 苹 果 CEO 蒂 姆 : 库 克 今日 就 苹果 地 图 存在 诸多 功能 问题 向 用 户 发 函 致 区 ， 称 未 能 为 用 户 提供 世界 一 流 的 
产品 体验 感到 抱 菊 。 这 是 苹果 首次 回应 近来 备 受 诉 病 的 iOS 6 地 图 服务 。 4 上 周 ， 苹 果 对 此 事 唯一 给 出 的 评论 就 是 ，“ 我 们 还 处 于 起 步 阶 
P, 在 信 中 库 克 棕 到 ， 谷 歌 地 图 是 jiOS 条 统 中 的 第 一 款 地 图 ， 但 是 ' 随 着 时 间 的 推移 ， 苹 果 想 为 客户 提供 更 好 的 地 图 功能 ,包括 .. 
苹果 流 媒 体 电台 服务 遇 障 碍 : 索尼 /ATV 对 价格 不 满 


网 易 科技 讯 9 月 29 日 消息 ， 据 国外 媒体 报道 ， 苹 果 的 互联 网 电台 服务 计划 最 近 在 获得 由 索尼 /ATV 控 制 的 音乐 的 授权 方面 远 到 因 难 ， 索尼 
人 ATV 是 此 前 由 迈克 尔 - 杰 克 逊 (Michael Jackson) 和 索尼 合资 成 立 的 企业 ， 最 近 成 为 了 全 球 最 大 育 乐 发 行商 。 苹果 与 索尼 /ATV 的 争议 在 


F, 芋 果 在 其 服务 上 流 媒体 播放 索尼 ATV 的 歌曲 应 该 支付 的 价格 ， 三 目前 的 情况 是 , 苹果 需要 与 全 部 音乐 公司 进行 U 
库 克 暗示 苹果 将 推出 革命 性 电视 机 产品 


网 易 科 技 讯 12 月 7 日 消息 ， 据 国外 媒体 报道 ,苹果 CEO 蒂 姆 : 库 克 ( Tim Cook ) 今日 在 接受 NBC 采 访 时 表示 ， 公 司 的 下 一 重大 计划 将 是 
有 关 家 庭 中 的 客厅 部 分 一 一 瞳 示 苹果 将 很 可 能 推出 革命 性 的 电视 机 产品 。“ 当 我 走 进 客 厅 ， 并 打开 电视 ， 我 感觉 时 间 像 是 倒退 了 20 到 30 


网 易 科技 讯 12 月 7 日 消息 ， 据 国外 媒体 报道 ， 苹果 二 星 周 四 重 回 硅谷 法 庭 。 二 星 希 望 联 邦 法 官能 推翻 或 修改 此 前 做 出 的 罚金 高 达 10.5 亿 
美元 专利 案 判 决 。 今年 8 月 由 陪审 团 做 出 的 裁决 显然 没 能 符合 两 家 公司 的 意愿 ， 二 星 苹 果 以 及 他 们 的 律师 希望 法 官 高 兰 囊 对 此 进行 更 
改 。 在 有 周 四 的 听证 会 上 ， 两 家 公司 了 预计 会 围绕 赔偿 金额 于 发 争执 ， 高 达 10.5 亿 美元 的 损失 赔偿 是 凑 只 产权 类 案件 中 判 赔 金 额 最 高 的 一 
笔 。 — 


11 条 记录 1/3 页 下 一 页 12 3 


图 14-15 全 文 搜索 分 页 效果 


14.5.4 ”实现 天 键 词 高 完 显 示 


接 下 来 将 继续 完善 前 面 创建 的 全 文 搜索 引擎 。 其 中 关键 词 高 党 能 够 让 用 户 始 终 清楚 目 己 
所 搜索 的 词汇 ， 从 而 提高 搜索 体验 。 要 实现 高 完 字 显示 ， 符 见 的 方式 有 两 种 : 一 种 是 使 用 
PHP EWJ, z PHE Sphinx API 提供 的 BuildExcerpts 方法 实现 。 正 则 符 换 相信 读者 
己 经 非常 熟悉 ， 所 以 这 里 将 使 用 BuildExcerpts 方法 实现 。 下 面 首 先 对 BuildExcerpts 方法 做 
一 个 简单 的 介绍 。 

BuildExcerpts 方法 用 于 对 Sphinx 搜索 结 采 生成 数据 摘要 及 标识 局 完 字 。 如 果 操 作成 功 
将 返回 true, de WOR lol false。 方 法 格式 如 下 。 

function BuildExcerpts ( $docs, $index, $words, $opts=array() ) 

BuildExcerpts 方法 共 文 持 4 个 参数 ， 其 中 参数 docs 220 kO e IG A 
类 型 )， 参 数 index 表示 Sphinx 索引 数据 库 名 称 ; 参数 words 表示 需要 处 理 的 关键 词 ， 人 参数 
opts 用 于 设置 高 之 学 处 理 参数 。opts 参数 使 用 关联 数组 表示 ， 设 置 或 改变 关联 数组 的 值 ， 将 
决定 高 亮 显示 的 方式 。opts 共 文 持 8 个 选项 ， 分 别 如 下 。 

> before match, 在 匹配 的 关键 词 前 添加 的 HTML 或 字符 。 

> after match: 在 匹配 的 关键 词 后 添加 的 HTML 或 学 符 。 

> chunk separator: 摘要 之 间 的 分 隔 符 ， 默 认为 “.…”。 
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limit: 摘要 最 大 包含 的 字符 ， 黑 认为 256。 
around: 每 个 天 键 词 块 左右 选取 的 词 的 数目 ， 鸭 认为 5。 
exact phrase: 是 否 仅 高 亮 精 确 瑟 配 的 整个 查询 词组 ， 默 认为 false。 
single passage: 和 是否 抽取 最 佳 的 一 个 段 洲 ， 黑 认为 false。 
weight order: 是 人 否 对 抽取 的 段 沙 进 行 排序 ， 默 认为 false。 
opts 数组 有 许多 选项 ， 对 于 实现 关键 词 高 有 党 显 示 而 言 ， 只 逢 要 设置 before match 及 
after match 项 即 可 。 接 下 来 将 在 原 有 的 功能 上 实现 关键 字 高 亮 显示 功能 ， 代 但 如 下 所 示 。 
as 


* 搜索 返回 数据 


x Enter description here ... 
a 
public function search (){ 
C ("LAYOUT ON", false); 
// 导 入 系统 扩展 函数 库 
SEET 
(mney Ter ore | tee 
SEET ganinxadl", WW w powo) z 


VW WW V 


Şsp=new SphinxClient(); 
SEConn=Ssp->setServer("192.168.2.16" ,9312); 


“worde ür ldecode m EE 

LE (1! $ras = $sp=>cuery ( Swords, “=W )) 1 
die ( "Brrors™ SE Ee () )? 
exit (0); 


} 

Tapore Ore UI 

SPade = new Page (Şras["total"],4); 

$show = $Page->show();// 分 页 显示 输出 
Şsp->setLimits (Ş$Page->firstRow,ŞPage->listRows); 


Srs = $sp=>cquery ( Swords, "=w ) 5 


EE 

sst = null; 

/ /提取 key 

foreach ( S$strs as Skey => Svalue ) { 
SSE C S UU beer 

| 

// 得 到 所 有 文章 id 

article d= trim O eet Te 

// 查 询 MySQL 数据 库 


Sarticle=M ("Article") z 
a a EH 
Srows=$article=>where (Scara) =>select() p 


/ / RETAS 

$opts = array (); 

$opts['before_match'] = '<b><font color="red">'; 
$opts['after_match'] = '</font></b>'; 


foreach ($rows as $key=>$value){ 
$title[] = $rows[$key] ['title']; 
$content[] = preg_replace ('/\[[\/]? (b|img|url|color|s|hr|pļ|list|iļalign| 
emailļ|u|font|code|hide|table|tr|td|thļ|attach|list|indent|float).*\]/','',strip_tags 
($rows[$key] ['content'])); 
} 
$title=$sp->BuildExcerpts ($title,"main" ,Swords ,SoPts) ; 
$content=$sp->BuildExcerpts ($content, "main", $words ,$opts) ; 
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foreach (Srows as $k=>$v){ 
Srows [Sk] ['title'] = $title[$k]; 
$rows[$k]['contnet'] = $content[$k]; 
} 
search ino > a a Ee EE EE Ee Kees 
re 
EO Es 
SEnis >assion ("oae", SSnow)s 
Sthnis- >assicon("list", $rows); 
} 
Sthis=>display() s 


y—| 一 下 


} 
W ERIS HTR, PREIEI A KI ERE T R e AR. TARR 14-16 
EZR. 
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首页 (cl 搜索 中 心 | 发 表 日 志 
苹果 全 文 搜索 


关键 字 为 【苹果 】 共 有 11 条 记录 ， 用 时 0.153 秒 


库 多 就 革 果 地 图 缺陷 向 用 户 道歉 ` 未 提供 一 流 


网 易 科技 讯 9 月 28 日 晚间 消息 ， 苹 果 CEO 蒂 姆 - 库 克 今日 就 苹果 地 图 存在 诸多 功能 问题 向 用 户 发 函 致 欢 ， 称 未 能 为 用 户 提供 世界 一 流 的 
产品 体验 感到 抱 坎 。 这 是 苹果 首次 回应 近来 备 受 诉 病 的 iOS 6 地 图 服务 。 4 上 周 ， 苹 果 对 此 事 唯 给 出 的 评论 就 是 ， 我 们 还 处 于 起 步 阶 
EB, 在 信 中 库 克 提 到 ， 谷 歌 地 图 是 IOS 系 统 中 的 第 一 款 地 图 ， 但 是 ' 随 关 时 间 的 推移 ， 蔷 订 想 为 客户 提供 葛 好 的 地 图 功能 ,包括 .. 
苹果 流 媒体 电台 服务 遇 障 碍 : 索尼 /ATV 对 价格 不 满 


网 易 科 技 讯 9 月 29 日 消息 ， 据 国外 媒体 报道 ， 苹果 的 互联 网 电台 服务 计划 最 近 在 获得 由 索尼 /ATV 控 制 的 音乐 的 檬 权 方面 殉 到 困难 ， 索 尼 
/IATV 是 此 前 由 迈克 尔 :杰克 还 (Michael Jackson) 和 索尼 合资 成 立 的 企业 ， 最 近 成 为 了 全 球 最 大 音乐 发 行商 。 苹果 与 索尼 /ATV 的 争议 在 


于 ， 苹果 在 其 服务 上 流 媒 体 播放 索 尼 /ATV 的 歌曲 应 该 支付 的 价格 ， 但 目前 的 情况 是 ,苹果 这 要 与 全 部 音乐 公司 进行 
库 克 暗示 苹果 将 推出 革命 性 电视 机 产品 


图 14-16 KEFR 


全 此 ， 一 个 和 干 万 级 别 的 全 文 搜索 引擎 就 开发 完成 了 。 该 者 可 以 在 此 基础 上 继续 完善 ， 例 
如 增加 关键 子 联 想 提 示 、 关 键 子 目 动 完成 、 结 来 绥 存 等 实用 功能 。 


14.6 小结 


本 革 首 先 对 全 文 搜索 概念 进行 了 评 细 介绍 ， 帮 助 读者 认识 全 文 搜 索 与 普通 搜索 的 区 别 。 
接 看 介绍 了 目前 最 主流 的 Sphinx 全 文 搜 索引 人 擎 ， 通 过 这 些 内 容 的 学 习 ， 和 帮助 读者 加 深 对 
Sphinx 的 认识 。 然 后 分 别 介 绍 了 Sphinx 中 文 分 词 发 行 厂 Coreseek Wik. FEMNA T 
Coreseek 管理 工具 的 使 用 以 及 款 引 源 配 置 等 。 通 过 这 些 内 容 的 和 学习， 读者 完全 可 以 掌握 中 文 
检索 的 实战 应 用 。 

最 后 本 章 通过 Sphinx 官方 提供 的 API， 结 合 ThinkPHP MVC 框架 ， 实 现 了 一 个 性 能 高 
效 的 全 文 搜索 引擎 。 本 章 内 容 比较 票 凑 ， 但 实战 性 比较 强 ， 访 者 如 果 需 要 彻底 掌握 本 章 内 
容 ， 务 必 按 照 内 容 编排 进行 学 习 及 实验 。 
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内 容 提 要 


MongoDB 是 非 关 系 型 数据 库 中 最 像 关 系 型 的 数据 库 。 它 不 仅 支 持 大 型 数据 库 中 的 许 
多 特性 ， 还 保持 了 NoSQL 数据 库 中 特有 的 快速 、 简 单 、 易 用 特点 。 本 章 将 从 最 基础 的 非 
关系 型 数据 库 开 始 介 绍 ， 帮 助 读 者 轻松 理解 NoSQL 各 种 概念 ， 例 如 库 概念 、 集 合 概 念 、 
文档 概念 等 。 

针对 MongoDB， 本 章 将 从 最 基础 的 Bson 语法 开始 讲解 ， 因 为 Bson 是 MongoDB 唯一 
支持 的 命令 行 操 作 语 言 ， 读 者 只 有 在 完全 理解 Bson 后 ， 才 能 在 实际 项 目 开发 中 得 心 应 手 。 
本 章 最 后 还 结合 MVC 框架 ， 详 细 介绍 在 MVC 中 MongoDB 的 CURD 操作 。 


学 习 目 标 


了 解 NoSQL 的 概念 。 

掌握 MongoDB 在 各 平台 下 的 安装 。 

掌握 Bson 语言 。 

掌握 MongoDB 的 高 级 查询 。 

掌握 MongoDB 的 性 能 优化 。 

掌握 MongoDB 配合 PHP MVC 的 实战 开发 。 
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15.1 MongoDB 介绍 


MongoDB 是 一 套 高 性 能 、 易 开发 的 文档 型 数据 库 。 它 使 用 键 值 对 形式 存放 数据 ， 能 够 
存放 包括 字符 串 、 数 组 、 数 据 厅 列 、 图 片 、 视 频 等 在 内 的 大 多 数 数据 文档 。MongoDB 完善 
的 设计 ， 高 效 的 可 编程 性 使 其 成 为 当前 NoSQL 产品 中 最 热门 的 一 种 。 在 对 MongoDB 进行 
介绍 前 ， 首 先 有 必要 来 了 解 NoSQL. 

1. NoSQL 概述 

NoSQL 全 称 为 Not Only SQL， 指 的 是 非 关 系 型 数据 库 。 传 统 的 关系 型 数据 库 是 基于 
SQL 语言 进行 操作 的 ， 虽 然 在 功能 上 是 够 强大 ， 但 由 于 一 些 应 用 在 局 部 上 只 需要 简单 的 数据 
操作 ， 传 统 的 关系 型 数据 库 显得 效 座 低 下 。 随 看 Web 2.0 的 到 来 ，SNS 应 用 的 兴起 ， 对 数据 
库 性 能 提出 了 苛刻 的 要 求 ， 关 系 型 数据 库 也 烘 露 出 其 难以 解决 的 问题 。NoSQL 由 于 其 本 身 
的 特点 迅速 得 到 发 展 。 

非 基 系 型 数据 库 与 天 系 型 数据 库 最 大 的 区 别 是 处 理 馆 辑 上 的 不 同 ， 但 使 用 的 目的 是 相同 
的 《都 是 用 于 存放 数据 )。 非 天 系 型 数据 库 使 用 数据 集 存放 数据 ， 在 数据 操作 上 没有 统一 的 
操作 语言 ， 如 果 读 者 使 用 过 Memcache 的 话 ， 那 么 对 非 关 系 型 数据 库 并 不 会 感到 阳 生 ， 因 为 
Memcache 驶 属于 非 基 系 型 数据 库 的 稚 形 。 

在 使 用 方式 上 ， 非 关系 型 数据 库 使 用 键 值 对 存放 方式 。 但 数据 库 毕 葛 首 先 要 有 库 的 概 
念 ， 才 能 称 为 数据 库 ， 例 如 数据 类 型 、CURD 操作 、 用 户 权 限 、 数 据 索 引 、 数 据 备 份 、 数 据 
同步 等 ， 这 些 都 是 数据 库 所 共有 的 概念 。NoSQL 同样 具备 这 些 概念 ， 甚 至 在 功能 上 更 加 丰 
府 ， 这 一 点 在 后 面 的 学 习 过 程 中 读者 将 会 体会 到 。 

随 看 非 天 系 型 数据 库 的 发 展 ， 有 些 成 熟 的 非 关 系 型 数据 库 完全 可 以 代 人 蕉 传统 的 缓存 服务 
髓 、 队 列 服务 右 等 基于 键 值 对 的 数据 服务 对 象 。 这 也 是 NoSQL 日 后 发 展 的 一 个 方 问 。 

2. NoSQL 的 特点 

NoSQL 有 许多 特点 ， 但 最 显著 的 就 是 性 能 得 到 大 幅 提 升 。 相 对 于 传统 的 关系 型 数据 
库 ， 在 处 理 大 并 发 及 大 数据 量 时 ，NoSQL 具有 不 可 替代 的 优势 。 下 面 将 分 别 介绍 。 

(1) 处 理 海量 数据 

非 关 系 型 数据 库 是 处 理 海 量 数据 的 最 佳 产 品 。 对 于 SNS 应 用 而 言 ， 瞬 间 的 数据 爆发 是 
难免 的 ， 传 统 的 关系 型 数据 库 在 使 用 表 分 区 、 服 务 器 集群 等 技术 后 虽然 也 能 解决 部 分 问题 。 
但 是 当 单 一 表 数 据 量 达到 上 亿 条 记录 时 ， 数 据 库 引 擎 下 难 以 有 效 地 对 数据 进行 处 理 。 非 关系 
型 数据 库 在 查询 数据 时 ， 不 存在 复杂 的 关系 运算 ， 每 个 集合 都 有 各 自 的 唯一 属性 ，NoSQL 
在 碍 询 数据 集合 时 ， 残 如 同和 直接 读 取 文本 ， 而 不 必 进 行 复杂 的 SQL 计算 与 转换 ， 所 以 有 效 
地 提高 了 查询 效率 。 

(2) 高 可 用 性 和 蜗 可 扩展 性 

NoSQL 虽然 是 一 个 很 广泛 的 名 称 ， 没 有 特 指 菏 一 产品 。 但 几乎 所 有 主流 的 NoSQL 数据 
库 都 提供 了 高 可 用 性 和 高 可 扩展 性 的 特性 ， 这 一 点 相对 海量 型 数据 的 SNS 应 用 而 言 是 尤其 
香 要 的 。 传 统 的 关系 型 数据 库 通常 是 基于 单 点 服务 来 提供 运算 的 ， 一 旦 中 途 需 要 增加 服务 市 
点 ， 就 必须 停止 数据 库 服 务 器 才能 实现 。NoSQL 数据 库 通 常 都 提供 了 简单 高 效 的 数据 同 


422 D E 


第 15 章 使 用 MongoDB 


步 、 贡 点 热切 换 等 功能 ， 从 而 真正 实现 高 可 用 性 及 高 可 扩展 性 。 
(3) 数据 处 理 灵 活 高 效 
传统 的 关系 型 数据 库 对 数据 的 处 理 必须 提供 条 理 清晰 、 结 构 分 明 、 类 型 精确 的 源 数 据 ， 
虽然 这 些 结构 对 控制 数据 的 完整 性 及 可 靠 性 是 非常 有 必要 的 ， 但 对 于 一 些 碎片 式 的 数据 而 
言 ， 显 然 灵 活 上 自由 的 存储 方式 更 加 高 效 。 
前 面 介 绍 的 就 是 NoSQL 数据 库 中 3 个 最 显著 的 特点 ， 但 NoSQL 并 非 只 有 优点 ， 它 也 
存在 缺点 ， 并 且 是 明显 的 。 归 纳 如 下 。 
> NoSQL 没有 特定 的 查询 语言 ， 也 不 兼容 SQL 结构 化 标准 查询 语言 。NoSQL 在 各 三 
商 产 品 中 都 有 各 上 自 的 NoSQL 文 持 语言 ， 这 些 语言 不 能 通用 。 
> 正如 前 面 所 言 NoSQL 对 网 站 局 部 数据 操作 提供 了 民 好 的 文 持 ， 但 并 不 表示 NoSQL 
适合 网 站 的 全 部 应 用 范畴， 也 并 不 表示 NoSQL 可 以 代替 关系 型 数据 库 。 例 如 事务 性 
较 高 的 、 需 要 多 表 复 杂 碍 询 的 以 及 传统 的 智能 商务 型 网 站 ， 并 不 适合 选用 NoSQL. 
> NoSQL 是 为 解决 大 数据 存储 而 生 的 ， 相 对 于 关系 型 数据 库 ， 其 还 不 够 成 熟 。 此 外 
NoSQL 数据 库 还 没有 比较 完善 的 商业 化 支持 。 
3. MongoDB 特点 
MongoDB 是 NoSQL 数据 库 中 比较 著名 的 一 个 ， 其 最 大 的 特点 是 特性 丰富 、 功 能 强大 、 
使 用 简单 。MongoDB 官方 一 直 标 榜 其 为 最 像 关 系 型 数据 库 的 非 关 系 型 数据 库 ， 事 实 上 也 是 
如 此 。MongoDB 为 了 方便 初学 者 及 开发 人 员 能 迅速 上 手 ， 许 多 管理 终端 命令 借鉴 或 二 接 兼 
容 MySQL 命令 ， 这 对 于 PHP 开发 人 员 而 言 是 非常 友好 的 。 
MongoDB 使 用 BSON 数据 格式 作为 数据 存放 结构 ， BSON 数据 格式 就 是 Json 数据 格 
式 的 二 进 制 化 ， 对 于 开发 者 而 言 ， 在 使 用 及 表现 形式 上 两 者 是 一 致 的 。 也 就 是 说 开发 人 员 可 
以 使 用 熟悉 的 Json 对 MongoDB 所 有 数据 结构 进行 操作 ， 例 如 数据 集 〈 表 ) 的 增 、 删 、 改 、 
二 
MongoDB 松散 的 数据 结构 ， 丰 宇 了 数据 特性 及 后 期 扩展 ， 并 且 能 够 文 持 关系 型 数据 库 
中 的 主键 DD、 数据 唯一 性 、 数 据 索 引 、 数 据 校 验 和 等。 具体 的 特性 如 下 。 
> 模式 自由 。 支 持 动态 查询 、 完 全 索引 ， 可 以 方便 地 对 文档 中 内 骨 的 对 象 及 数组 进行 
> 面 问 文档 储存 。 开 发 人 员 可 以 将 现 有 的 序列 文档 直接 导入 到 MongoDB 数据 集中 。 
> 高 效 的 储存 机 制 。MongoDB 对 数据 并 没有 特殊 的 限制 ， 任 何 数据 〈 包 括 大 型 的 图 
片 、 视 频 等 ) 保存 到 MongoDB 数据 库 中 ， 最 终 都 将 被 转换 为 二 进 制 数据 ， 从 而 在 一 
定 程 度 上 保证 读 写 的 高 效 ， 以 及 确保 数据 完整 、 安 全 等 。 
> 支持 复制 和 故障 恢复 。 提 供 了 主 - 从 、 主 - 主 模式 的 数据 复制 及 服务 器 之 间 的 数据 


复制 。 
> 目 动 分 户 并 文 持 云 级 别 的 伸缩 性 。 文 持 水 平 的 数据 库 集 群 ， 可 以 动态 添加 额外 的 服 
"e, 


> 支持 Python、PHP、Ruby、Java、C、C#、JavaScript、Perl 及 C++ 语言 的 驱动 程序 。 
> 宛 全 文 持 分 布 式 架构 。 

> Sphinx 1.x 开始 文 持 MongoDB 驱动 。 

前 面 对 NoSQL 及 MongoDB 进行 了 深入 的 介绍 。 接 下 来 将 结合 示例 代码 ， 重 点 介绍 
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MongoDB 的 实战 应 用 ， 帮 助 读者 轻松 开发 珊 效 、 敏 捷 的 互联 网 应 用 。 下 和 面 首先 介 绍 
MongoDB 的 安装 。 


15.2 MongoDB 的 安装 


与 大 多 数 开源 软件 一 样 ，MongoDB 同样 也 分 为 Windows 平台 安装 及 Linux 平台 安装 。 
Windows 平台 主要 用 于 开发 及 测试 环境 ， 而 Linux 平台 能 够 用 于 测试 及 生产 环境 。 接 下 来 将 
分 别 介绍 这 两 种 平台 上 的 MongoDB 安装 过 程 。 


15.2.1 在 Windows 下 安装 MongoDB 


在 Windows 下 安装 MongoDB 相对 比较 人 简单。 整体 共 分 为 3 个 步骤 : 首先 下 载 对 应 平台 
的 MongoDB 源 代 人 码 包 ; 然后 解压 MongoDB 源 代码 包 ， 在 命令 行 终端 中 切换 到 MongoDB 
源 代码 所 在 的 bin 目录 ， 直 接 局 动 mongod.exe 可 执行 文件 ， 最 后 将 MongoDB 作为 Windows 
随机 局 动 服务 即 可 。 下 男根 据 流程 进行 安 疙 。 

(1) 下 载 MongoDB Join 

读者 可 以 在 http://www.mongodb.org/downloads 网 址 中 找到 相应 平台 的 MongoDB 源 代码 
包 ， 如 图 15-1 所 示 。 


0S X 64-bit Linux 32-bit Linux64-btt Windows 32-bit Windows 64-bit Solaris GA D Source 
note mote 


| Production Release (Recommended) 


222 

1 download download download tgz 
download download download 

Changelog *legacy-staic *legacy-static "20023 zip 

Release Motes 

Nightty download download download tgz 
download download download 

Changelog "legacy-static *legacy-static =2008R2+ zip 

Previous Release 

208 

11192012 download download download tgz 
download download download 

Changelog *legagy-static *legacy-static WU zip 

Release Notes 

Nighithy download download download tgz 
download download download 

Changelog *legagy-static *legacy-static *2008R2+ zip 


图 15-1 MongoDB 源 代码 包 


这 里 选择 “Windows 32-bit” 平 台 ，MongoDB 版 本 为 2.0.8。 读 者 可 根据 实际 环境 目 行 选 
择 。 下 载 完 成 后 将 得 到 “mongodb-win32-i386-2.0.8.zip” 源 代码 包 ， 将 该 压缩 包 解 压 到 合适 
的 目录 ， 并 将 目录 名 称 改 为 mongodb。 整 个 mongodb 目录 及 文件 结构 如 下 。 


mongodb 
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bin/ 
bsondump .exe ee bson 备份 工具 
mongo.exe ‘seese mongodb 管理 终端 
mongod.exe “ee mongodb 主 服务 
mongodump .exe “ee 备份 工具 
mongoexport .exe “ee 数据 导出 工具 
mongofiles .exe ee GidFs 管理 工具 
mongoimport .exe tessere... 数据 导入 工具 
mongorestore .exe ee 数据 恢复 工具 (支持 大 数据 ) 
MONgOS .exe ee. 分 片 程序 
mongostat.exe “ee 综合 性 能 监控 工具 
mopnegokop, ege Te 恋 写 能 力 监 控 工 具 


GNU-AGPL-3.0 
README 
THIRD-PARTY-NOTICES 
MongoDB 下 载 后 ， 不 需要 安装 ， 和 直接 运行 由 可 。 在 运行 mongod.exe 程序 之 前 ， 首 先 需 
要 创建 logs 及 data 目录 ， 分 别 用 于 存放 MongoDB 日 志文 件 及 数据 库 文件 ， 完 成 后 束 可 以 局 
动 MongoDB J。 
SH o 
打开 命令 行 终端 ， 并 切换 到 MongoDB 源 代码 所 在 的 bin 目录 ， 运 行 以 下 命令 启动 


MongoDB 7 
D:\mongodb\bin>mongod.exe --dbpath=D:\mongodb\data --logpath=D:\mongodb\ logs\ 


mongodb. Log 


命令 运行 后 ，MongoDB WKH J, WKI 15-2 所 示 。 


vn 管理 吕 : f 令 提示 符 - mongod.exe --dbpath=D:\mongodb\data --logpath=D:\mongodb\logs\mongodb.l... ~ 0 eo 


D:\mongodb\bin>mongod.exe --dbpath=D:\mongodb\data --logpath=D:\mongodb\logs\mongodb.1og EE 
all output going to: D:\mongodb\logs\mongodb.1og 


图 15-2 启动 MongoDB 主 服 务 


(3) 加 入 Windows E 
为 了 方便 下 次 局 动 ， 最 后 只 需要 将 mongod.exe 加 入 Windows 随机 服务 即 可 ， 命 令 
ks 


D:\mongodb\bin>mongod.exe --dbpath=D:\mongodb\data --logpath=D:\mongodb\logs\ 


mongodb.log -install 


D:\mongodb\bin>net start mongodb 


命令 成 功 运行 后 ， 打 开 服 务 管理 面板 ， 可 以 看 到 MongoDB 服务 已 经 成 为 Windows 系统 
服务 ， 如 图 15-3 所 示 。 
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# 


Mongo DB ER WE 状 弯 END SRA = 
S Microsoft Software Shadow Copy P.. 管理 岩 影 复 抽 服务 制作 的 基于 ,.. 手动 本地 系统 

停止 此 服务 hongo DB Mongo DB Server 

重启 动 此 服务 È Mozilla Maintenance Service Mozilla ERR RESE... 手动 本 地 系统 
$ Multimedia Class Scheduler 基于 系统 范围 内 的 性 务 优先 级 ,… 自动 RE SE 


图 15-3 MongoDB 服务 


至 此 ，Windows 下 的 MongoDB 就 安 闭 完成 
T , 现在 可 以 使 用 mongo.exe 终端 管理 工具 直接 登 A 管理 吕 ; 命令 提示 符 - mongo.e... 
ES MongoDB 数据 库 了 ， 如 图 15-4 Hi To D:\mongodb\bin>mongo.exe 


MongoDB shell version: 2.0.8 
connecting to: test 


> 


15.2.2 在 Linux 下 安装 MongoDB 


MongoDB 给 人 最 直接 的 印象 是 使 用 简单 ， 包 
AER. Œ Linux 平台 上 同样 如 此 ， 下 面 图 15-4 管理 MongoDB 数据 库 
将 详细 介绍 CentOS 6.0 操作 系统 下 的 安装 过 程 。 

(1) 下 载 MongoDB 

打开 http://www.mongodb.org/downloads 页 面 ， 选 择 相 应 平台 的 Linux 安装 源 代 人 码 包 。 这 
里 选择 的 是 “Linux 32-bit” 平 台 ，MongoDB 版 本 为 2.0.8。 也 可 以 直接 在 命令 终端 下 载 。 

[root@~]# cd /datal 


[root@bogon datal]# wget http://fastdl.mongodb.org/linux/mongodb-linux-i686-2.0.8. 
tgz 

下 载 完 成 后 ， 只 需要 解压 即 可 。 

[rootebogon datal]# tar zxvf mongodb-linux-i686-2.0.8.tgz 

[root@bogon datal]# cd mongodb-linux-i686-2.0.8 

[root@bogon mongodb-linux-i686-2.0.8]# dir 

bin GNU-AGPL-3.0 README THIRD-PARTY-NOTICES 

为 了 方便 管理 ， 这 里 将 mongodb-linux-i686-2.0.8 目录 下 的 文件 复制 到 /usr/local/mongodb 
目录 中 ， 过 程 如 下 。 

[root@bogon datal]# cd /datal 

[root@bogon datal]# mkdir -p /usr/local/mongodb 

[root@bogon datal]# cp -R mongodb-linux-i686-2.0.8/* /usr/local/mongodb/ 

[root@bogon datal]# cd /usr/local/mongodb/ 

与 Windows WA "P, Œ Linux 下 同样 不 需要 对 MongodDB 进行 编译 ， 只 需要 直接 运 
ÍT bin 目录 下 的 mongod 主 程序 即 可 。 但 在 这 之 前 首先 需要 创建 日 六 及 数据 库存 放 目 录 。 


[root@bogon mongodb]# mkdir logs 


[root@bogon mongodb]# mkdir data 
[root@bogon mongodb]# Les 
bin data GNU-AGPL-3.0 logs README THIRD-PARTY-NOTICES 


[root@bogon mongodb]# cd bin 
完成 前 面 的 步骤 ， 接 下 来 就 可 以 司 动 了 。 
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(2) 局 动 服务 
司 动 MongoDB 服务 的 过 程 与 Windows 平台 几乎 是 一 样 的 ， 只 需要 修改 路 径 形 式 即 可 。 


[root@~]#./mongod --dbpath=/usr/local/mongodb/data/ --logpath=/usr/local/ mongodb/ 


logs/mongodb.log --fork 
上 述 命令 中 ， 加 入 了 fork 选项 ， 访 选项 表示 将 mongod 服务 进程 推送 到 后 台 运 行 。 
mongod 局 动 程序 还 文 持 其 他 局 动 可 选项 ， 分 别 如 下 。 


> 


VW WW WW WW WM WW WM WWW D D 


--dbpath: 指定 数据 库 目 录 。 

--port: 指定 通信 端口 ， 默 认 是 27017。 

--bind_ip: 绑 定 管理 ip。 

--directoryperdb: 使 用 独立 的 文件 夹 存 放 数 据 库 文 件 。 

--logpath: 指定 日 志和 存放 目录 。 

--logappend: JE HETEKIA GEJ mi) 

--pidfilepath: 指定 进程 文件 路 径 ， 为 空 则 不 产生 进程 文件 。 
--keyFile: 集群 模式 的 关键 标识 。 

--cpu: 周期 性 地 显示 CPU 和 IO 的 使 用 率 。 

--journal: 局 用 日 六。 

--ipv6: JHH IPV6 文 持 。 

--nssize: 指定 .ns 文件 的 大 小 ， 以 MB 为 单位 ， 默 认 16MB ， 最 大 2GB。 
--maxConns: 最 大 的 并 发 连接 数 。 

--notablescan: 不 允许 进行 表 扫 摘 。 

--quota: 限制 每 个 数据 库 的 文件 个 数 ， 默 认为 8 个 。 

--quotaFiles: 每 个 数据 库 的 文件 个 数 ， 需 要 配合 一 quota 参数 使 用 。 
--noprealloc: 关闭 数据 文件 的 预 分 配 功 能 。 

--auth: 以 用 户 授 权 模 式 局 动 。 


MongoDB 成 功 启 动 后 ， 默 认 情 况 下 会 使 用 27017 端口 作为 通信 端口 。 通 过 命令 可 以 检 
训 到 相应 的 服务 状态 。 


[root@bogon bin]# netstat -ant lp |grep 27017 


tcp 0 0 0.0.0.0:27017 0.0.0.0:* LISTEN 15643/mongod 


(3) 加 入 随机 局 动 
成 功 局 动 后 ， 只 需要 将 mongod 主 程序 加 入 rc.local 配置 文件 中 即 可 。 为 了 系统 重新 局 
动 时 局 动 MongoDB 数据 库 服务 ， 首 先 打 开 rc.local 配置 文件 。 


[root@bogon bin]# vi /etc/rc.local 


然后 在 该 配置 文件 最 后 面 加 入 mongod 启动 项 ， 代 码 如 下 所 示 。 
#!/bin/sh 
i Uais EE be ekecucec! wartcer™ all che ocher inite EE 


# You can put your own initialization stuff in here if you don't 
# want to do the full Sys V style init stuff. 
touch /var/lock/subsys/local 


/usr/local/mongodb/bin/mongod --dbpath=/usr/local/mongodb/data/ --logpath=/usr/ 


local/mongodb/logs/mongodb.log --fork 


保存 rc.local 配置 文件 ， 全 此 MongoDB 数据 库 就 安装 完成 了 。 现 在 就 可 以 使 用 mongo 
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终端 管理 工具 登录 MongoDB 数据 库 了 。 


[root@bogon bin]# ./mongo 


MongoDB shell version: 2.0.8 


connecting to: test 


15.3 MongoDB 的 使 用 


初次 接触 MongoDB 的 读者 通常 会 觉得 MongoDB 数据 操作 语法 很 凌乱 ， 没 有 SQL 语言 
那样 结构 清晰 、 迎 辑 严 课 。 事 实 上 ，MongoDB 的 数据 库 操 作 是 非常 直观 的 ， 无 论 开 发 者 进 
行 多 么 复杂 的 数据 操作 ， 始 终 面 癌 的 是 文档 模型 ， 文 档 中 的 任意 属性 或 字段 都 可 作为 操作 的 
条 件 。 在 操作 语言 上 就 是 普通 的 JavaScript， 所 以 只 要 理解 MongoDB 的 数据 结构 ， 束 能 轻松 
要 驭 数据 库 的 第 见 操作 。 


15.3.1 理解 MongoDB 的 数据 结构 


MongoDB 数据 结构 是 由 数据 库 (database), RA (collection)、 文 档 (document) 三 部 
分 组 成 的 。 其 中 ， 开 发 人 员 主 要 面 对 的 是 colection 及 document 。 其 中 数据 文档 
(document) 是 数据 的 载体 ， 其 数据 格式 定义 如 下 代码 所 示 。 

{文档 名 :文档 内 容 } 

如 上 述 代码 所 示 ， 其 中 文档 名 称 是 必 选 项 ， 文 档 内 容 文 持 JavaScript ém d, 
例如 数组 、 对 象 、 字 符 串 等 《详细 的 数据 类 型 将 在 下 一 节 进 行 介 绍 )。 数 据 文 档 配 合 操作 方 
法 ， 共 同 组 成 MongoDB 数据 库 操作 语句 。 一 条 了 最 简单 的 MongoDB 操作 语句 如 下 。 

> db.c1.save ({user: {username:"ceiba",email:"kf@86055.com"}}); 

相信 读者 对 上 述 代码 并 不 会 感觉 阳 生 ， 从 代码 格式 上 与 常见 的 Json 是 类 似 的 ， 这 种 数 
据 格式 在 MongoDB 中 称 为 Bson (Binary Json). HP cl 表示 和 集合 名 称 (AEK); save 表 
示 操 作 方 法 ;操作 方法 中 的 参数 即 是 标准 的 Json 数据 。 格 式 如 图 15-5 所 示 。 


db. cl save({user: {username: ceiba”, email: kf@86055. com }1): 


0 


集合 名 操作 方法 文档 名 称 文档 内 容 
图 15-5 MongoDB 数据 格式 


其 中 文档 相当 于 关系 型 数据 库 中 的 字段 列 ， 文 档 名 称 是 为 了 方便 数据 管理 而 定义 的 分 类 
名 称 。 同 一 个 数据 集合 中 ， 人 允许 多 篇 文档 同名 。 在 MongoDB 中 ， 文 档 没 有 字段 的 概念 ， 一 
篇 文 档 中 的 数据 格式 是 松散 的 ， 同 名 文档 之 间 不 要 求 数 据 类 型 相同 ， 也 不 要 求 数 据 项 目 相 
同 。 如 以 下 代码 所 示 。 

> db.cl.save({user:["ceiba", "kf@86055.com"]}); 

上 述 代 人 码 使 用 数组 作为 文档 内 容 ， 在 文档 归 类 上 属于 user 文档 。 从 上 述 示例 中 可 以 看 
H, MongoDB 整个 数据 结构 核心 由 文档 组 成 ， 而 文档 束 是 Bson。 数 据 库 是 MongoDB 的 数 
据 存 放 单 元 ， 数 据 集合 相当 于 文档 分 类 ， 而 文档 是 整个 MongoDB 数据 结构 中 最 基础 的 单 
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位 。 如 图 15-6 所 示 。 

MongoDB 之 所 以 称 为 最 像 关 系 型 数据 库 的 非 
关系 型 数据 库 ， 就 是 因为 MongoDB 也 遵循 关系 
型 数据 库 中 的 操作 语法 。 前 者 使 用 Bson 语法 ， 后 
者 使 用 SQL 语法 。 对 于 习惯 了 天 系 型 数据 库 的 读 
者 ， 接 下 来 将 通过 表 形 式 ， 将 MongoDB 与 关系 
型 数据 库 进 行 对 比 ， 以 便 让 该 者 更 深刻 理解 两 者 | 
之 间 的 逻辑 结构 ， 如 表 15-1 所 未 。 


数据 库 1 


database 1 


集合 2 
collection 2 


集合 1 
collection 1 


SS 15-1 MongoDB 与 MySQL 结构 对 比 


MongoDB MySQL 
databases 《数据 库 ) databases 〈 数 据 库 ) 
collection (RE) tables 《数据 表 ) 
document 〔 文 档 ) row 〈 列 ) 
document name 《文档 名 称 ) 全 图 15-6 MongoDB 数据 结构 


值得 注意 的 是 ， 在 定义 数据 文档 时 ， 文 档 内 容 虽 然 可 以 无 限 层次 丛 套 ， 但 是 为 了 方便 
API 调用 ， 文 档 内 容 原 则 上 不 要 超出 两 层 数据 结构 ， 如 以 下 代码 所 未。 


db.cl.save({user: {username:"ceibas",password:123,introduce:{age:25,school:{Elementa 
ry:"gz86",Middle:"gz86"}}}}); 


上 述 代 人 码 虽 然 是 合法 的 数据 ， 在 JavaScript 中 调用 没有 问题 。 但 是 由 于 MongoDB 对 各 
语言 API 的 文 持 程度 不 同 ， 过 多 的 层次 网 人 套 将 让 数据 操作 变 得 困难 。 


15.3.2 ”数据 库 管理 


默认 情况 下 ，MongoDB 安装 完成 后 ， 会 自动 生成 test 数据 库 。 与 关系 型 数据 库 一 样 ， 
MongoDB 同样 支持 常见 的 数据 库 操 作 ， 例 如 显示 数据 库 、 创 建 数据 库 、 删 除数 据 库 、 修 复 
数据 库 等 。 详 细 的 命令 如 表 15-2 所 示 。 


SS 15-2 MongoDB 数据 库 管 理 命 令 


命 ZS 说 W 

Ser éi He OH 

show dbs 显示 数据 库 列 表 

show collections 显示 当前 数据 库 中 的 集合 ， 兼 容 命令 为 show tables 

use<db name> 切换 数据 库 ， 如 果 不 存 在 则 创建 

db.dropDatabase() 删除 当前 数据 库 

db.cloneDatabase("host") 远程 复制 数据 库 

db.copyDatabase("mydb","temp","host") 复制 数据 库 

db.getName() 碍 看 当前 数据 库 

db.statsO) 显示 MongoDB 状态 

db.repairDatabase() 修复 当前 数据 库 

db. version? 显示 MongoDB 版 本 信息 
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CZE) 
go S 说 明 
数据 库 常规 管理 

db.getMongo() 显示 当前 MongoDB 通信 IP 
db.help0) 显示 数据 库 操作 帮助 信息 

数据 集 管 理 
db.<table name>.helpO 显示 指定 数据 集 帮 助 信息 
db.createCollection0); 创建 数据 集 
db.<table name: drop) 删除 数据 集 
db.getCollection("name"); 获取 指定 数据 集 信 息 〈 包 括 索 引 、 容 量 等 ) 
db.getCollectionNames() 显示 MongoDB 所 有 数据 集 信息 
db.printCollectionStats() 输出 所 有 数据 集 详细 信息 (包括 索引 、 容 量 等 ) 
db.<table name>. Stats() 显示 指定 集合 的 详细 信息 
db.<table name>.count() 统计 指定 集合 文档 数量 
db. <table name>.dataSize() 查看 指定 集合 空间 使 用 大 小 

BI: P A8 
db.addUser("name'",”password”",true) 添加 用 户 ， 共 文 持 3 个 参数 ， 最 后 一 个 参数 表示 是 否 只 该 
show users 显示 所 有 用 户 
db.removeUser("userName") 删除 指定 用 户 


如 表 15-2 PR, MongoDB 并 有 提供 数据 库 创 建 命令 ， 但 提供 了 use 命令 ， 该 命令 用 于 
切换 数据 库 ， 如 有 果 切 换 的 数据 库 不 存在 ， 则 目 动 创建 。 默 认 情 况 下 ， 局 动 mongo 管理 终 
iio AIREA test 数据 库 ， 如 以 下 代码 所 示 。 

[root@bogon bin]# ./mongo 

MongoDB shell version: 2.0.8 

connecting to: test 

> show dbs 

local (empty) 

test 0.0625GB 


>> 

在 mongo 管理 终 山 中， 所 有 命令 都 使 用 “;” 作 为 命令 结束 符 。 同 时 ， 文 持 使 用 
JavaScript 语法 作为 MongoDB 操作 语法 ， 如 以 下 代码 所 示 。 

El Lr) 1 


. db.c1.save ({user:{username:"ceiba_"+i,password:i}}) 


Ee 
上 述 代 人 码 使 用 了 JavaScript ENAKI T Set ich A. GD R a MongoDB 
的 使 用 ， 更 多 的 语法 将 在 后 面 的 内 容 中 进行 介绍 。 


15.3.3 ”文档 数据 类 型 

前 面 提 到 过 MongoDB 文 持 币 见 的 数据 类 型 。 当 然 ， 由 于 开发 语言 乙 间 的 差异 ， 在 API 
调用 时 数据 奖 型 的 声明 方式 也 有 上 所 不 同 。 下 面 将 以 mongo 终端 管理 工具 为 例 ， 分 别 介 绍 
MongoDB 所 支持 的 数据 类 型 。 
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(1) null 

null 表示 空 值 或 不 存在 的 字段 ， 使 用 方式 如 下 。 

{name:null} 

(2) 布尔 值 类 型 

布尔 值 类 型 可 接受 的 值 包括 true 或 false， 使 用 方式 如 下 。 

{male:true} 

(3) 整数 

包括 32 位 及 64 位 的 整数 。 由 于 mongo 管理 终端 是 基于 JavaScript 编写 的 ， 所 以 在 声明 
64 位 整数 时 ， 管 理 终端 将 会 统一 转换 为 32 位 。 但 在 Java 等 API 上 则 不 存在 此 问题 。 使 用 方 
SR 

Lage: 20) 

RS EN 

包括 32 位 及 64 位 的 浮 点 数 。 特 性 与 整数 类 型 一 样 ， 使 用 方式 如 下 。 

(pay:80:12)} 

(5) 字符 串 

字符 串 是 MongoDB 中 最 常用 的 数据 类 型 ， 在 使 用 时 需要 加 双 引 号 。 文 持 UTF-8 编码 的 
字符 串 ， 使 用 方式 如 下 。 

{email:”kf086055.com”} 

(6) ObjectID 类 型 

默认 情况 下 系统 会 为 每 篇 文档 自动 生成 12 位 ObjectD， 该 字符 串 中 包含 了 时 间 戳 、 
机 堪 识 别 码 、 计 数 融 等 信息 。 碍 询 对 应 文档 ， 将 得 到 该 文档 对 应 的 ObjectD， 如 以 下 代 
码 所 示 。 

{"_id" : ObjectId("4df2dcec2cdcd20936a8b817")} 

(7) 日 期 类 型 

当前 日 期 及 时 间 ， 使 用 方式 如 下 。 

{user: {addtime:new Date ()}} 

(8) 正则 表达 式 

LFF JavaScript 正则 表达 式 作 为 售 询 条 件 ， 使 用 方式 如 下 。 

{"user.username":/ceiba/i} 

(9) ARAS 

GUTE A JavaScript (R, HFA TF. 

{"code":function test (){/*alert ('hello mongodb"')*/}}) 

(C10) 数组 

人 允许 特 接 插 入 JavaScript 数组 ， 使 用 方式 如 下 。 

db.cl.save({1list:["a","b",["c","d"]]}); 

d1) UR vi 

直接 导入 现 有 的 Json 文档 ， 使 用 方式 如 下 。 


{email:”kf@86055.com”} 
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15.3.4 插入 数据 


前 面 的 内 容 中 ， 己 经 简单 地 涉及 数据 集合 的 插入 操作 。 在 插入 数据 时 通常 使 用 的 操作 方 
法 有 insert 及 save。 这 两 个 方法 如 果 忽 略 文档 主键 的 话 在 使 用 方式 上 是 一 致 的 ， 但 在 一 些 场 
合 下 ， 主 键 ID 作为 唯一 索引 卫 ， 是 具有 特殊 意义 及 作用 的 ， 下 面 首 先 介 绍 主键 ID。 

(1) 文档 主键 ID 

默认 情况 下 ， 在 插入 数据 时 ， 系 统 都 为 每 篇 文档 分 配 一 个 主键 ID MongoDB 使 用 
“id” 键 作为 文档 的 主键 ， 系 统 使 用 ObjectID 数据 类 型 生成 序列 写作 为 主键 ID 值 ， 并 日 为 
该 键 值 生成 唯一 索引 。ObjectD 数据 类 型 是 一 种 经 过 序列 化 的 字符 串 ， 能 够 确保 每 次 生成 的 
唯一 性 《〈 非 目 动 增长 )。 如 以 下 代码 所 示 。 


> Ob. ci. Lnsert(iusernamer"Tceibait, age 25))? 
EE 
{ Ww TeW Ee EE e EE Ee Kee H e EE e EE LE 2 25 } 


前 面 提 到 过 ，MongoDB 文档 字段 是 不 分 数据 类 型 的 ， 包 括 主 键 ID。 开 发 人 员 完 全 可 以 
使 用 易于 理解 的 整数 类 型 作为 主键 ID， 如 以 下 代码 所 示 。 


> db.c4.insert ({_id:1,username:"ceiba",age:25});}; 


> db.c4.insert ({_id:2,username:"ceiba",age:26}); 

> db.c4.find(); 

ENEE E E EE Ee 
EH ecma A :260 


E S 

当然 ， 为 了 严 课 及 数据 的 唯一 性 ， 建 议 读者 使 用 默认 的 ObjectID 或 整数 类 型 作为 主键 
ID 值 ， 尽 量 不 要 使 用 字符 串 或 数组 一 次 的 数据 类 型 。 

(2) save 操作 方法 

save 操作 方法 是 用 于 回 数据 集合 插入 或 更 新 文档 的 一 个 音 用 方法 。 当 插入 的 文档 主 
键 ID 存在 冲突 时 ，save 方法 将 执行 数据 更 新 操作 ， 反 之 执行 数据 插入 操作 。 如 以 下 代码 
所 示 。 


> Ob. ODI. Seveal(l lel 00L, userneme: Yecslba -agas 257) e 


> db.c5.save({_id:002,username:"ceiba",age:25});}; 
> Clos CS TLN) S 


Eh ee EE E DE .5 
el 2 EE EE EE Ae 2200 
> db.c5.save({_id:001,username:"ceiba",age:24}); 
> olo. c5. ELME) 

(om l username t Yee irbat, Magen k 2A 
EE EE EE 0 uee rban kagen: 

> 


如 上 述 代码 所 示 ， 标 粗 的 即 为 save 产生 冲突 的 操作 。 由 于 _id 值 已 经 存在 ， 所 以 产生 了 
冲突 ， 这 种 冲突 的 处 理 方 式 古 执行 数据 修改 操作 。 

(3) insert 操作 方法 

与 save 操作 方法 一 样 ，insert 方法 也 是 问 集合 中 插入 数据 文档 的 一 个 党 用 方法 。 不 同 之 
处 在 于 inset 不 执行 冲突 处 理 。 也 就 是 说 insert 遇 到 冲突 时 直接 中 止 后 续 操 作 ， 而 不 是 执行 
修改 操作 。 如 以 下 代码 所 示 。 


> db.c5.insert ({_id:001,username:"ceiba",age:24});}; 
11000 cuplicace key error Lnoex? test. €5.95 icl. á Chjo keys 4 3 1.0 } 
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如 上 述 代 码 所 示 ， 由 于 主键 ID 为 001 的 文档 已 经 存在 ， 再 执行 insert 数据 插入 操作 
时 ，MongoDB 将 百 接 反馈 出 错 信 息 ， 并 终止 执行 。 


15.3.5 ”查询 数据 


伍 询 数据 无 论 是 在 非 关 系 型 数据 库 还 是 在 关系 型 数据 库 都 是 CURD 操作 中 比较 重要 的 
操作 。 在 关系 型 数据 库 中 ， 查 询 数 据 通 党 使 用 SQL 语言 ， 考 量 一 个 数据 库 的 完善 及 强大 程 
上 度 ， 很 大 原因 上 取 雇 于 该 关系 型 数据 库 对 SQL 标准 的 文 持 情况 。 同 样 ， 在 MongoDB 中 ， 要 
得 询 文档 数据 同样 要 遵循 MongoDB 上 所 文 持 的 操作 语言 。MongoDB 虽然 没有 统一 的 标准 化 
AWA. (H MongoDB 使 用 Bson 作为 数据 库 操 作 语 言 ， 并 且 使 用 find 操作 方法 作为 数据 
得 询 方 法 。 只 有 操作 方法 与 操作 语言 结合 一 起 使 用 ， 才 能 实现 数据 得 询 。 下 面 将 结合 示例 ， 
评 细 介绍 find 操作 方法 的 使 用 。 

Gd) 普通 查询 

在 MongoDB 中 进行 数据 查询 是 非常 形象 和 和 直观 的 。 一 条 最 简单 的 查询 语句 格式 如 下 。 

db.<table name>.find(); 

上 述 命令 将 返回 对 应 数据 集 的 所 有 数据 记录 ， 如 以 下 代码 所 示 。 


db.c2.find(); 


> 

{we Ee Ee le EE RE ee Kee H ee E 3 Weeibat, Wage 2 20 )} 
{we Ee ae ee E E ee Kee ee EN ee EE E E LE 3 21 } 
Ee ee e WEE Ee E 

{ "_id" : ObjectId("50b0464ec0d6flef5d6abbef"), "username" : "ceiba", "age" : 25 } 
> 


在 天 系 型 数据 库 中 但 询 数 据 时 ， 通 常 都 配合 where RERI, KMA RA 
询 。 同 样 ， 在 MongoDB 中 也 能 使 用 类 似 的 功能 ， 如 以 下 代 人 码 所 示 。 


> db.c2.find({age:25}); 
{ Ww OW s Obiecrtd(nbhObü/AëGäecdertletädpGabtbetfnh, Tusernameff a "Tceibai, Wagen Zä! 


上 述 代 码 相 当 于 以 下 SQL 语句 。 

select * from c2 where age=25 

可 以 看 到 ， 在 MongoDB 中 查询 数据 实际 上 就 相当 于 Bson 文档 属性 匹配 。 例 如 
“age:253” 可 以 理解 为 “age=25”。 表 示 在 cl 数据 集中 查询 age 字段 值 等 于 25 的 文档 。 如 果 
多 条 记录 符合 查询 要 求 ，find 操作 方法 将 返回 所 有 记录 。 

上 述 代码 实现 了 Bson 文档 简单 的 查询 。 在 一 些 复杂 的 Bson 文档 中 ， 例 如 多 层 Bson 的 
文档 ， 则 需要 使 用 “.” 分 隔 符 实 现 条 件 匹配 。 例 如 cp 数据 集 记 录 如 下 。 
> db.c6.find(); 


Ee BEE le DEE Ree LS 人 ES 
Een El 

{ wW LCU Ee ee Ke ER d EE E Ee E LW, 
WOE 

Ehe Ke E ke EE ee EE DN VusenrY 3 { MSSrneme’ 5 Veeusa 2+. 
Taoeft 2 I 

{ w e Dee Ehe DEE ebe EE EE EE EE EE EE E 
EE EE E 

{ Ww MEW g OoJect iic EE ee L54960; E E EE EH e E EE 
e LE EE 

{ Ww Te EE EE SSE RE s Weeioa 54, 
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E EE és 

E EE DEE UE SEE 6", 
WarereSw GI 

Ee Mee EE ee E e E E e Ee RE E E Elek Fee Re De TW, 
VE E ek eg e 

{ Ww e 8 OQoJect Ee HEN Re E EE SET 
vaget kS 

{ Ww iCY s OoJect irice EE Re E Vaser g { Wusernama s Weceiba 9w, 
TIGEN E: Oon 


现在 需要 查询 user 文档 中 age 等 于 5 的 记录 ， 代 人 码 如 下 。 

> DEE EE agen :5)) 

{ Ww e AE EE SE DIER Ree e ve 1 Wusernamet s Weeda E 
"age" : 5 } } 


如 上 述 代 码 所 示 ， 由 于 age 字段 隶属 于 use 字段 ， 所 以 在 查询 时 需要 使 用 “.” 分 隔 
符 。 查 询 其 他 层级 的 Bson 数据 时 ， 依 次 类 推 即 可 。 

(2) 使 用 JavaScript 

使 用 find 操作 方法 能 够 直接 对 多 条 数据 进行 过 历 ， 返 回 的 数据 结构 是 泛 型 的 。 事 实 上 
mongo 管理 终 疹 是 一 个 基于 JavaScript 编写 的 Shell, Hr DAt PJ DA A I E A iim E EH 
JavaScript Až WA FRET o 


> var rows =db.c6.find(); 
> while (rows.hasNext () ) { 


. printjson (rows.next () ) ; 


ks 


EE 


SE a 
EE EE EE 
Taoeit OU 


Ee Ee 


WEE EE H 
EE ET Ee Eeer, 
Tagen Ek 


w ICW Ee Ree 


El ka 
ee s Yceriba 2v. 
PACE 


Ga OoeeElaelt Ee 


EE E 
EE EE EE EE 
Ve EE 
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EE 


WE EH e 
EE RE 3 veeloa 4w, 
vage: A 


wW ICW 8 QoJect iriad ("S500050 EST 530C0aLloab 9ce") ; 


WEE E dek 
EE Ee e 
E 


EE 


TSSEA SE 
E EE e EE OO"; 
EGG a 6 


Ee Ee 


SST 
WEE EE s Vosloe yw, 
E Ne 


EE 


EE lt 
EE s Yceiba 9w, 
H CEN a G 


W ICU ss ODJeEcTt E EE Re )， 


vasem s 
WusernameY 3 El 
vgen O 


} 


当然 也 可 以 将 结果 转换 为 数组 ， 然 后 使 用 游标 输出 。 如 以 下 代码 所 示 。 


> var rows=db.c6.find() .toArray ();，; 


> rows[2]; 


{ 
EE 


SS 
EE TE s Vesna 2%, 
Ee FE 
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15.3.6 ”更 新 数据 


在 关系 型 数据 库 中 ， 通 党 使 用 update 关键 字 更 新 数据 。 在 MongoDB 中 同样 提供 了 
update 操作 方法 ， 用 于 更 新 MongoDB 数据 文档 。 使 用 格式 如 下 。 

update (criteria, objNew, upsert, multi); 

update 操作 方法 共 文 持 4 个 参数 ， 分 别 如 下 。 

> criteria: 更 新 操作 条 件 ，Bson 格式 。 

> objNew: 更 新 后 的 值 ，Bson 格式 。 

> upsert: 如 果 记 录 存 在 时 ， 采 取 更 新 还 是 插入 方式 。 默 认为 0， 即 更 新 方式 。 

> multi: 是 个 更 新 全 部 符合 操作 条 件 的 记录 。 默 认为 0， 即 只 更 狐 1 条 记录 。 

需要 注意 的 是 ， 如 采 数 据 集中 不 存 更 新 条 件 中 的 多 条 记录 ， 但 multi 参数 为 1 时 ， 将 会 
报错 并 停止 执行 。 下 面 结合 示例 演示 update 操作 方法 的 使 用 。 假 设 c4 数据 集结 构 如 以 下 代 


公所 示 。 
Se .fing(). 
i username EE i magerion) 
EEN Ser namen Ee ee 
de een EE EE 26 
> 


现在 需要 对 id 等 于 1 WAHIE, IER TZENA “username” FRAN 
“ceibas” 即 可 ， 代 人 码 如 下 所 示 。 


> db.c4.update ({_id:1}, {username:"ceibas"},0,1); 


> db.c4.find(); 

rb De EE EE e BEE) 

Ee RER EE EE 

Ee EE EE EE oe 260 
(Ame :USenmame Eed 
> 


这 里 的 更 新 条 件 并 非 限于 主键 ID 。 事 实 上 只 要 是 Bson 对 象 ， 都 可 作为 更 新 条 件 ， 如 以 
下 代码 所 示 。 


> db.c4.update ({username:"ceibas"}, {username:"ceiba"},0,1); 

细心 的 读者 已 经 发 现在 前 面 所 介绍 的 save 操作 中 已 经 涉及 数据 更 新 操作 。 事 实 上 save 
Wi update 的 别名 ， 所 以 在 使 用 方式 上 更 加 简单 ， 如 以 下 代码 所 示 。 

eeh E ick DEE E RER E Ee a ema EE Ee 


SE ELMO) E 
{ "_id" : 1, "username" : "ceiba", "email" : "kfR86055.com" } 


Es 


save 操作 方法 在 前 面 的 内 容 中 已 经 多 次 涉及 ， 这 里 不 再 细 述 。 在 MongoDB 中 ， 还 可 以 
使 用 魔术 变量 实现 数据 更 新 ， 关 于 厅 木 变量 的 使 用 将 在 本 章 后 面 进行 介绍 。 


15.3.7 删除 数据 


remove 操作 方法 用 于 删除 数据 集合 中 的 Bson 数据 。 在 功能 实现 上 ，remove 可 以 实现 批 
星 删 除 以 及 根据 Bson 条 件 进行 删除 。 无 论 是 执行 哪 种 删除 操作 ， 在 MongoDB 中 都 是 单 问 
不 可 地 的 (没有 询问 提示 )。remove 使 用 格式 如 下 。 
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qb .<table name> .remove ({ 删 除 条 件 } ) ，; 
其 中 删除 条 件 是 一 条 Bson 语句 。 由 于 文档 主键 在 生成 时 默认 会 自动 创建 主键 索引 ， 所 
以 尽量 使 用 文档 主键 ID 作为 删除 条 件 。 如 以 下 代码 所 示 。 


SE ee Se EE 

E e d e EE EE EE E DEE Com en 
Ee ee Cenova Ee Ee te 

> 


需要 注意 的 是 ， 删 除 条 件 并 非 必 需 的 。 如 果 删 除 条 件 为 衬 ， 系 统 将 执行 数据 集 清空 操 
作 ， 类 似 于 MySQL 中 的 TRUNCATE 操作 。 如 以 下 代码 所 示 。 


> db.cl.remove (); 
> olos ls ELRES 
> 


15.4 条件 操作 


前 面 已 经 介绍 过 ， 在 数据 集 CURD 操作 中 ，Bson 格式 中 的 “:” 可 以 理解 为 “=”。 使 用 
数据 库 的 好 处 就 是 方便 存储 及 碍 询 数据 ， 非 关系 型 数据 库 也 不 例外 。 而 要 实现 方便 储存 及 奉 
询 数据 ， 提 供 完善 的 条 件 操作 是 必 不 可 少 的 ， 显 然 仅 仅 使 用 “=” 条 件 操作 是 不 能 达到 需求 
的 。 事 实 上 MongoDB 已 经 提供 了 类 似 于 关系 型 数据 库 完善 及 强大 的 条 件 操作 ， 包 括 条 件 判 
断 、 模 式 匹 配 、 上 映射 但 询 等 。 接 下 来 将 分 别 介绍 。 


15.4.1 条 件 判断 语句 


在 关系 型 数据 库 中 ， 开 发 人 员 可 以 使 用 直观 的 比较 运算 表达 陈 实 现 数据 的 条 件 判 断 。 例 
如 “where idy3” 而 在 MongoDB 中 ， 由 于 使 用 Bson 作为 操作 语言 ， 显 然 SQL 中 的 这 些 比 
bit d EMi MongoDB 的 。 为 此 ，MonsgoDB 使 用 魔术 变量 实现 表达 式 符 号 符 
换 ， 分 别 为 大 于 $get (>)、 小 于 ($I1D、 大 于 或 等 于 〈$gte) 等 。 相 信 谈 者 对 这 些 魔术 变量 并 不 
会 感到 阳 生 ， 因 为 本 书 前 面 介 绍 的 ThinkPHP RIIE, Smarty 模板 引 获 束 是 使 用 类 似 的 表 
达 式 符号 。 下 面 首 先 介绍 魔术 变量 的 使 用 方式 。 

表面 分 别 介 绍 了 大 于 、 小 于 、 大 于 或 等 于 比较 表达 式 魔 术 变 量 。 其 中 等 值 表 达 式 可 以 直 
接 使 用 Bson 表示 ， 如 以 下 代码 所 示 。 


> db.c8.find({_id:2}); 
但 使 用 魔术 变量 时 ， 比 较 表 达 陈 将 变 为 如 下 格式 。 
qb .<table name>. find ( (“RTR : (ERRE : "RHE" }}); 
其 中 ， 操 作 方 法 find 也 可 以 是 其 他 的 操作 方法 ， 例 如 update, remove 等 。 接 下 来 将 结 
合 示例 代 但 ， 演 示 表 达 式 的 使 用 。 假 设 c8 数据 集 记 录 如 以 下 代码 所 示 。 


Se Ee 


Ee E EE E le EE Ee 
E EE EE E 
ee WER EE Dt E Ee E 
EE E EE EE EE Det KEE 
EE E D E :G0 
(cu Seormame ee Det EES 
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dE E SEET 
E ENEE EE H 0 CY :000 
(Gl 8 EIERE EE E EE 00 
dE EE EIERE Tceiba on, agoen LL J 


接 下 来 需 要 查询 id 大 于 5 的 记录 ， 代 但 如 Sg 


> db.c8.find({_id:{$gt:5}}); 

le C Senmnamen: Cen O7 Vece s e 
eeh EE EE EE e E A CY e O 
lee EE EE ae 0 
ee 2, Tusernameft s Dceiba On, Tage LL J 


Er Gm, id 小 于 2 的 记录 ， 代码 如 下 所 示 。 


db.c8.remove ({_id: {$1t:2}}); 
ellome Sm ELME A 


> 

> 

(i sername kiN E EE E EE A) 
Tr EE E E E Ee EE o) 

UU ` " e UU " e UU A UU UU UU e 

ly E EE E A GER EE 
(Gl EE E EE EE 
(el G e eet A er 
EE E e e EE SO 
EE EE Een EE) 
Ee 9, "username" "ceiba_ 2i "age" TESS) 


同样 的 操作 也 适用 于 meng Save 等 操作 方法 。 需要 注意 的 是 ， 在 进行 表达 式 操 作 时 ， 
同样 需要 遵循 MongoDB 的 数据 类 型 限制 。 


15.4.2 $all 匹配 全 部 
$all 魔术 变量 用 于 匹配 文档 内 容 类 型 为 数组 的 数据 ， 如 以 下 代码 所 示 。 


EE KEE 

EE EE E 
db.c9.find({a: {$all:[0,1,2]}}); 

E EE EE ek 


需要 注意 的 是 ，$all 在 匹配 时 ， 必 须要 确保 被 匹配 的 文档 内 容 中 包含 有 匹配 条 件 中 的 所 
有 数组 元 素 ， 合 则 将 停止 匹配 ， 如 下 代 但 所 示 。 


el lee 


VM 


Oh en Mi WM 
Ti 


如 上 述 代码 所 示 ， 虽然 a 文档 内 容 中 存在 匹配 条 件 中 的 “1、2、3” 数 组 元 素 。 但 是 由 
于 不 包含 匹配 元 素 “11”， 所 以 最 终结 果 为 空 


15.4.3 $exists 检查 字段 


$exists 用 于 判断 匹配 条 件 中 的 文档 属性 是 否 存在 或 不 存在 。 如 果 条 件 成 过， 将 执行 操作 
方法 ， 否 则 将 放弃 执行 。$exists 只 文 持 布尔 值 〈 即 true 或 false)， 如 以 下 代码 所 示 。 


> db.c9.find({a: {$exists:true}}); 

E Bh (Oe E EE AE 

CES 

如 上 述 代 码 所 示 ， 由 于 a 字段 存在 ， 所 以 find 操作 方法 能 够 顺利 执行 。 但 是 ， 如 果 改 变 
$exists 变量 值 ，find 操作 方法 将 会 得 到 相反 的 结果 ， 如 以 下 代码 所 示 。 


> db.c9.find({a: {$exists:false}}); 
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同样 的 原理 ， 如 果 比 较 一 个 不 存在 的 文档 字段 ， 得 到 的 结果 与 $exists:true 一 样 ， 如 以 下 


代码 所 示 。 
> db.c9.find ({aa: {$exists:false}}); 
人 
E E Se EE 
> 


利用 $exists 代码 ， 可 以 方便 地 实现 CURD 操作 前 的 各 种 检查 。 


15.4.4 null 空 值 处 理 
MongoDB 数据 集 允 许 使 用 null 空 值 作 为 文档 数据 类 型 。 如 以 下 代码 所 示 。 


EE CLIO. insere ({ussr; EE EE 

> Ob. ClO nsert(iuserriieceeiba 2Y- agez Lj) e 

> Clo. ClO. insere ("USS VesSiba SH agez EIER 

> Clo. CLO, EE {usr ESUoa AW Ages EE 

= Clo. ClO EE Se E Ee AE g 

> cls clo tinch) z 

{ U ict s QIeelELel(" SOULOoE64GEETESSESBTLUS55 Vuser s Wceiba Lw, Wage 2 20 ) 

{ WU ie s OSJect ic GER Ek e Ee EE EE E 3 Weeiba 2%; Wage s 21 ) 

{ WG s Object Iich Y50b0or269rG4cdler58Ce LOSY), VuserY s Woelloa SV, Wage 8 mull 

{ w Te s Object iic“ 50b0otSloredelers5segl056Y); EE 44W. vacev EE 

1 U TM s (QINSCELC E e E LOSC y Maser 9 Wosiba 5%- Vage gz WH 

如 上 述 代 人 码 所 示 ，ceiba_3 与 ceiba 4 及 ceiba_5 的 age 属性 数据 类 型 分 别 表示 三 种 数据 
类 型 ， 即 空 值 aull、 字 符 串 nul， 以 及 空 符号 。 接 下 来 将 获取 空 值 Cnull) 的 数据 ， 如 以 下 代 
公所 示 。 

> db.c10.find({age:null}); 

{ W ic 8 O5Jecc re ee EE SSH, Wage s mwili e 


上 述 代码 得 到 的 结果 是 正确 的 。 因 为 null 是 一 个 合法 的 数据 类 型 ， 所 以 允许 直接 操作 。 
而 “2 ”虽然 是 Bson 值 ， 但 这 里 只 表示 衬 字 人 符 串 值 ， 与 null 数据 类 型 并 不 同一 概念 。 

需要 注意 的 是 ， 在 MongoDB 2.0 之 前 ， 衬 值 运算 不 能 直接 赋值 进行 操作 。 可 以 使 用 
$exists 魔术 变量 进行 处 理 ， 如 以 下 代码 所 示 。 


Gel ege tages {Sins el E EE EE ER 
EE E ee >s Wceiba 3Y, Vage şs mull 1 
> 


15.4.5 $ne 比较 


在 SQL 语言 中 ， 使 用 “!=” 或 “<>” 作 为 非 〈 不 等 于 ) 比较 运算 符 。 在 MongoDB 中 ， 
同样 也 内 置 了 $ne 魔术 变量 ， 用 于 实现 “! =” 比 较 运算 。 使 用 方式 如 以 下 代码 所 示 。 


db.cl0.find({age: {$ne:null1}}); 


> 

Ce EE SCS1059"), “USSEY > Weeiba I", Yage™ s 20 } 

{ w id" s OQogJectirc(YS0o0bileoteddletrogeel0sa"); Vuser” 3 Weeiba 2"; Vage” 8 21 |) 

{ * 10" 8 OosSEEIG SO0505ESOEGAEHLEESSCSTLUSE AN) UUSEE s Weeriba AW, aoe s Hull } 
{ “W ic" 3 OQoJect rce ("S0o0brS Kee UUSEE 3 Weeiba 5"; ade g WW } 

> 


上 述 代码 表示 三 询 age 字段 不 等 于 空 值 的 数据 。 既 然 可 以 实现 查询 操作 ， 同 样 也 可 以 实 
IN, update 或 insert 等 操作 ， 如 以 下 代码 所 示 。 


O 回 439 


PHP MVC 开发 实战 


domen 
"iq" 
"iq" 
" jq" 


WERO kW 


domen 


TETAN 


ësst o 


: ObjectId("50b0bflebf64dlef58c8105a"), 
: ObjectId("50b0bf26bf64dlef58c8105b"), 


: ObjectId("50b0bf31bf64dlef58c8105c"), 


: ObjectId("50b0bf37bf64d1lef58c8105dqd"), 
qb .c10 .remove ({age: {$ne:null)}}); 


Q ELME e E 


: ObjectId("50b0bf26bf64dlef58c8105b"), 


15.4.6 $mod 取 模 运算 


RH SE, 


EE A 


"user" : 


"user" : 


"user" : 


Tceiba 24r Taoei 
ucelban suy "age" 
Yeerba An, Tagen :; 
veerbaro t Nage: 
weerbarst kaget 


KARRA, Æ PHP 中 通常 用 于 实现 验证 但 或 者 数据 分 页 等 功能 。 


EE 
2 Eck 


BEE 


~ 


wor } 


3 mull } 


在 


MongoDB 中 ， 可 以 利用 取 模 运算 ， 对 特定 的 文 梢 属性 实现 奇偶 数 得 询 ， 如 以 下 代码 所 未 。 


DV Ri SI 

> db.c8.find(); 

(el EE 
(oe Semmamer ne. 
(Gl :A Se 
(i :om EE 
EE TEE e 
e EE 
TA S mus ername wis 
(ol Sr namen i: 
> db.c8.find({age: {$mod: 
EE EE E EE EE 
er ETH 
E EE 
TAN 9- Wusername” s 


"ceiba_ 


"ceiba_ 


[2,1]}}); 


"ceiba_ 


FOSA 


WeeENSa 9w, 


"ceiba_ 


JEE 
El EE 
EE Eer Au 
EE 
oun 
EE 
Ss 
EE 


E 
veerba 5u; 


Taoef 
sageti: 
vagen: 
Taoef k 
Taoef 
Taoef 
vaget: 
sagon: 


"age" e 
"age" e 
"age" e 


UU age UU 


取 模 运 zm 中 使 用 数组 [2,1] 进 行 表示 。 
必需 要 取出 age 为 偶数 的 数据 ， 


RA. Jr 
代码 所 示 。 


E 


EE lye Ka 


TEKAN 


E a 


3 2 USermemer g 


EA e 


4, “username™ : 
See EE TEE 
8 


TETAY: EE EE 


[2,0]}}); 


"ceiba 


"ceiba_ 


15.4.7 $in, nin 枚 举 查 询 


魔术 变量 $in 用 于 实现 枚 举 操 作 ， 与 SQL 语言 中 的 in 作用 相关 似 。 文 持 匹 配 所 有 整数 类 
型 的 文档 字段 。 假 设 c8 数据 集 数据 记录 如 下 。 


lege 


WEE KZ 


一 一 一 一 一 一 一 一 V 


UU id UU 


Web Ke e 
METON e 
Web Ké e 


TETON e 


EE E e 


UU Sen Beta e 


Eeer 


; "username" 


2y Vugername s 
a Tusernameif s 
EE EE 


SEET 


EE s 


3 
4 
5 

3 6O Vuserneaeme” s 
7 
8, "username" : 
9 


EAN 
veerba 4u, 
ES 
We Ee E 


其 中 元 素 2 表示 取 模 的 操作 数 ; 


O ~ UI 


Le el ei sei ee ` Lee) 


} 
} 
} 


TEE) 


1 表示 整除 后 的 


只 需要 改变 取 模 运算 中 的 操作 数 即 可 ， 如 以 下 


tagen: 
vaget: 
vaget: 
vagen: 


"ceiba_ 


"ceiba_ 
veeiba ou, 


E 


Gw 
EE 
EE 
WEE A 
Ee 
Ge 


dale 
vagen: 
Taoef 
vagen: 
vaget: 
vagen: 
s T EOT 
Ce e kee 


UU age UU 
UU age UU 


O Oo ~ On OD A 


(e Lee ee see jee sc) 


现在 需 ;要 查询 age 字段 值 为 4、5、6 的 数据 ， 使 用 $in 魔术 变量 束 很 容易 实现 ， 如 以 下 
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代码 所 示 。 
he eeh (Tage; EE Ae el yy})? 
e EE EE EE e A 
:3 aS ername GER EE he 
ee EE EE EE E E EE 
> 
魔术 变量 $nin 与 $in 的 作用 是 相反 的 ， 但 使 用 方式 一 致 ， 如 以 下 代 人 得 所 示 。 
> lo- c8. he (ages el e Kei 
EE 0 EI E E EE E 
dE EEN EE E EE E 
EE EE EE KE 
ER EE E E EE E ES 
EE EE El 


15.4.8 for, $nor 判断 查询 
默认 情况 下 ，MongoDB 使 用 “and” 作 为 所 有 Bson AWARA, WA FRIR. 


> db.c8.find ({_id:2,age:4}); 


EE EE A} 


如 上 述 标 粗 代码 所 示 ， 在 Bson 查询 中 ， 共 文 持 两 个 操作 条 件 ， 分 别 为 _id:2 和 age:4。 
默认 情况 下 操作 条 件 之 间 的 关系 是 “与 ”关系 。 转 换 为 SQL 语句 ， 如 以 下 代码 所 示 。 


select * from c8 where '_id'=2 and 'age'=4 


MongoDB 提供 了 $or 及 $nor 魔术 变量 ， 用 于 改变 条 件 操作 的 运算 关系 。 其 中 $or 表示 
“或 、 或 者 ”$nor EIR “JER”. $or 及 $nor 的 使 用 方式 与 其 他 魔术 变量 有 些 区 别 ， 格 式 
如 下 

ës Te E EE esche SCT? 

可 以 看 到 ， 前 面 介 lo a 但 $or 或 $nor 需要 在 
条 件 操 作 语 句 前 进行 声明 ， 并 且 使 用 “[]” 包 含 操作 和 条件。 下面 将 结合 示例 代码 演示 $or 及 


$nor 的 使 用 。 
SEET 
(i Seormname EE age ak 
lé EIERE EE de Lagen el 
Cadu e A USernaamey .Veenmasd Ee 
E EE EE EE E 
de EH EE EE 
全 
Ee EE EEN E EE EE s I0 
E EE EE E 
> db.c8.find({$or:[{_id:2}, {age:5}]}); 
EE EE E :4 
(el 0 EE EE EE EE S] 
> 


如 上 述 标 粗 代码 所 示 ， 转 换 为 SQL 人 查询， 将 得 到 以 下 代码 。 
See Ee E Ee 


BS 的 使 用 方式 也 一 样 ， 但 作用 是 相反 的 ， 如 以 下 代 人 码 所 示 。 


eld. ego rial Smor ii 162}. {face5}]})? 


Tardu k A username E E E BA Vage 6 
(Sernamer EE ee} 
d E :6 Smamer EE E A 007) 
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Ee e EE EE GE Cm: KE 
OA CA a EE EE EE E 
EE Cerba non EE 


15.4.9 ”$type 映射 至 询 


为 了 便于 理解 Bson 数据 类 型 ， 提 高 开发 效率 ，MongoDB 提供 了 $type 魔术 变量 ， 用 于 
映射 MongoDB 上 所 文 持 的 Bson 数据 类 型 。 开 友人 员 只 需要 理解 $type 映射 值 即 可 ， 而 不 需要 
记 住 文 档 属 性 中 的 数据 类 型 ， 从 而 提高 查询 效率 。 如 表 15-3 ro, 

kR 15-3 BSON 数据 类 型 映射 


Objectid ObjectID 

Boolean 4 布尔 值 

Regular expression JavaScript 正则 表达 式 
JavaScript code JavaScript 代码 
Symbool 标记 

JavaScript code with scope JavaScript 代码 

32-bit integer 32 位 整数 

Timestamp Timestamp F} EJ ÆR 
64-bit integer 64 位 整数 


接 下 来 将 通过 代码 演示 $type 魔术 变量 的 使 用 。 假 设 c10 数据 集 文档 记录 如 以 下 代码 所 示 。 


GE e 


> 

EE EE Se EE e 
de EE Ee EE EE EE s muL ) 

Ee KEE e Gg Ee EE) E lp 27 3 4 513 

{ "_id" : ObjectId("50b1476472f8ca2d8551f77f"), "username" : "ceiba" } 

{oT OA oe Z 0 senmame :en 

(sl ObjectId("50b1478472f8ca2d8551f781"), "username" L19 } 


DV 在 需要 查询 username 为 null 的 数据 ， 在 映射 查询 中 ，$type 为 10 时 即 表示 null 值 。 
所 以 可 以 直接 使 用 $type:10 表达 式 查 询 空 值 数据 ， 如 以 下 代码 所 示 。 


> db.cl0.find({username: {$type:10}}); 
{ W TOW s Ee EE Sola Ae I2 roadsS I VW), Vmsernemet g mwili J 
> 


15.4.10 使 用 正则 表达 式 匹 配 


$regex 魔术 变量 ， 将 让 正则 匹配 查询 变 得 更 加 直观 。$regex 使 用 格式 如 下 。 
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{" 属 性 名 ",{Sregex:" 正 则 表达 式 ",$options:" 枚 举 值 "} } 
其 中 正则 表达 式 支 持 所 有 JavaScript 正则 表达 式 。$options 与 JavaScript 正则 处 理 中 的 
options 作用 是 一 样 〈 即 枚 举 有 效 值 的 。 接 下 来 将 通过 示例 代码， 演示 $regex 魔术 变量 的 使 
用 。 假 设 cl 数据 集 数 据 记 录 如 以 下 代码 所 示 。 


> ‘lo GL. ae 

{ “Y ice 3 QoSJjeetELle( 30514s237218Ga26835351178C ) "USE s “veeloa ll } 
{ Y iew 3 QoSJeetLle( S00l483072E8Ca208551E786N), SET s GELUOa 21W J 
{1 “Y iew 3 @QojJjectird (Y S00l4633721T8Ca208551E788"), Vuser" s “Celoa 3S1" } 
{ “ic EE ele e EE Vuser™ 9 Weceiba 41w } 
{ “Y ic 3 Oo]EcEICI SU5OL4S3O72ESCa2683531E730 ) Vuser™ 9 Weeiba Sl J 
{ Y ic" 3 @oJjectric("S0obol4esir /2E8Ca208551E791*), Vuser" 3 Yceiba 52" | 
ee ek E "UUSEE s Yeeiba 593" J 
{ Y idl" 3 QoJjeetle( "S00l4e4672E8Ca20835351E793"), Vuser" 9 Yeeilba 54W |) 
> 
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现在 需要 俘 询 user 属性 值 后 级 为 1 的 文档 。 使 用 正则 表达 式 即 可 轻松 实现 ， 如 以 下 代码 


所 示 。 
> db.c1.find ({user:{$regex:"ceiba_.*1$",$options:"i"}}); 
{| WwW ICW g OoJjecc iel (“50o 42372 GE EE g Wegioa Liw } 
{ WwW Te EE El Ree E EE E Elle 
Ree Een Ee EE Ee GE EEN 
ee EE e EE E EE E e lee 
{ w eW ee EE E Ee L790 Eeer E Ee 
> 


正则 表达 式 是 计算 机 语言 中 通用 的 一 套 字符 处 理 规则 ， 在 各 种 编程 语言 中 均 有 广泛 的 应 
用 。 在 MongoDB 中 运用 好 正则 匹配 ， 首 先 需 要 读者 掌握 并 熟悉 正则 表达 式 。 


15.4.11 


limit, skip 限制 查询 
在 MySQL 数据 库 查 询 中 ， 通 常 需要 使 用 limit 关键 字 限 制 查询 记录 范围 。 这 样 做 不 仅 能 


够 有 效 提高 查询 速度 ， 还 能 实现 数据 分 页 。MonsgoDB 同样 也 提供 了 limit 操作 方法 ， 该 方法 
用 于 限制 返回 结果 范围 ， 使 用 格式 如 下 。 


db. table name> .findq() .Limit (数量 ) ， 


接 下 来 将 通过 示例 代码， 演示 limit 操作 方法 的 使 用 。 假 设 cl 数据 集 共 有 10 条 记录 ， 


如 以 下 代码 所 示 。 
> clo ecl Een boer 
{ w EE EE y Vusert z Wiser LW, Wage z 1L ) 
{ w eW EE E p VVSG 3 Wuser 2w; Wage gz 2 1 
{ W ic 3 Object ic SOSTEE5950962560535965s52e0 VUSELW DSS 3Y, Wage 3 3 ) 
(5 人 EEC SOSEES559625639565S52C E g Wuser 4U- Waga E 
{ U ict s ODJectiic ("50blog raso96256996562024); Yusen s Vussr 5%, Wage s 5 ) 
{ U 16 s ObJect ic (s500lobogra5o96256996562034); EE Wage s 6 ) 
RER KEE ee ee e EE E s Wussr TW, Vage s 7 
EE EE EE WUSELY EE EE ; S ) 
lee EE EE EE EE EE E ) 
er EE EE EE s Vuser LOW, EE 
> 


接 下 来 将 使 用 limit 操作 方法 限制 返回 结 东 只 显示 前 3 条 记录 。 


> db.c1.find().limit (3); 
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EE EE EE 9 WUSET LW; Jade s 1 et 
EE EE E HEET SE EE EE 2w; Yaga s 2 J 
EE EE EE EE SEOE 3 3 


将 上 述 代 码 转 换 为 MySQL 查询 语句 ， 如 以 下 代码 所 示 。 

selece esom ©@l limige 0737 

细心 的 读者 已 经 发 现 ， 在 MongoDB 中 limit 方法 只 接受 1 2%. AMERA E aE H 
limit 操作 方法 是 不 能 实现 数据 分 页 的 。 假 设 需要 实现 以 下 SQL EWR. 

EE ENEE 


ER SQL 查询 表示 在 结果 中 取出 第 5 条 记录 ， 一 共 取 3 Z. MEE 6 一 8 条 之 间 
的 数据 记录 。 要 在 MongoDB 中 实现 上 述 查 询 操 作 ， 还 必须 借助 于 skip 操作 方法 。 如 以 下 代 


公所 示 。 
> db.cl1.find() .skip(5) .limit (3); 
{ W ict s Object ic EE VUSELW 3 Vuser 6Y, Wao s 6 ) 
{ W Te 3 EE Ee EE Se E EE a Wuser T4, Va s 7 ) 
{ W ict 3 Object ic EE VuserY UUSEE OV, Vage 3 Bl 


skip 操作 方法 用 于 实现 得 询 记录 跳 园 ， 只 接受 一 个 整数 类 型 参数 。limit 操作 方法 用 于 限 
制 返回 记录 条 数 ， 同 样 也 只 接受 一 个 整数 类 型 参数 。 这 两 者 的 配合 是 实现 数据 分 页 的 基础 
(先后 顺序 不 受 限 制 )。 


15.4.12 count 查询 记录 条 数 
count 操作 方法 用 于 统计 数据 查询 记录 。 支 持 与 其 他 操作 方法 配合 使 用 ， 格 式 如 下 。 


db .<table name>. [操作 方法 1] . [操作 方法 2] .count (1); 
其 中 操作 方法 是 可 选 的 查询 选项 ， 如 有 果 不 存 在 操作 方法 ，count 将 统计 当前 表 所 有 记 
录 。 如 以 下 代码 所 示 。 


El EE 
LO 


count 方法 文 持 一 个 可 选 参 数 ， 接 受 的 参数 值 为 0 AV) 或 1。 如 果 为 0 时 ， 表 示 忽 略 
count 操作 方法 前 的 所 有 操作 方法 (find 除外 )。 如 以 下 代码 所 示 。 


EE EE 
LO 


如 上 述 代码 所 示 ， 因 为 使 用 了 skip 及 limit 限制 查询 ， 理 论 上 第 5 条 记录 跳 转 到 第 8 条 
记录 共 相 隔 3 条 记录 〈 即 筛选 记录 )。 但 结果 却 显示 了 10 条 ， 这 显然 与 我 们 所 期 待 的 不 一 
样 。 接 下 来 将 通过 修改 count 操作 方法 的 参数 值 ， 改 变 count 统计 方式 。 如 以 下 代码 所 示 。 

> db.cl.find() .skip(5) .1imit (3) .count (1); 


3 
> 


结合 前 面 介绍 的 skip, limit 操作 方法 ， 至 此 数据 分 页 所 需要 的 3 个 关键 数据 记录 (起 始 
记录 、 限 制 记录 、 总 记录 ) 已 经 具备 ， 在 进行 数据 分 页 时 将 变 得 简单 。 


15.4.13 sort 查询 结果 排序 


结果 排序 主要 分 为 升序 及 降序 。 在 MySQL 等 关系 型 数据 库 查询 中 ， 通 常 使 用 “ORDER 
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BY” KFK Æ MongoDB 中 ， 结 果 排 序 也 是 使 用 Bson 实现 的 ， 但 需要 配合 sort 操作 
方法 使 用 。 格 式 如 下 。 

db .<table name>.find() .sort ({ 排 序 字 段 :11}) ; 

其 中 排序 字段 可 以 是 Bson 文档 中 的 所 有 合法 字段 或 属性 《一般 选择 数学 类 型 或 
ObjectD 次 型 的 属性 )， 排 序 字段 值 共 文 持 2 个 值 ， 分 别 为 -1《〈 倒 序 ) 及 1 升序)。 如 以 下 
代码 所 示 。 


Clos cl ere E o SOLT E E Ee ee 


E Ee EE EE EE s Wuser LW, EE g l J 
{ WwW Te ee Ee En E E ee E GE ER E E E z Z2 } 
EE EE EE EE EE SEE EE EE 
E EE EE EE 8 Wuser AW, Wage a 4 ) 
{ W 16 EE EE Vuser" 8 WISSGP SV Vage 5 5 ) 
{ W ic 8 Object iid (¥50blbsraso92625699656203Y);, Vuser" s Wuser 6Y, Wage s 6 ) 
{ W ic s Object iCY SUOOSTEE50962560535965s52620  WuserY UUSEE TY, el 
{ Y KEE EE Ee EE SE E EE s Wuser oW, Vaoe 3 8 ) 
{ W166 3 Object Iic Y 50blog raso96256990656206Y); WUSELW UUSEE OV, Wage 3 9 ) 
{ U ic s OsJect Irc" s500lobograso9625699C56207Y); Vusert s WUSer LOU, Wage s 10 ) 
> Clo cl PLNE) ee E e ter 

{ W 16 ; Object ic" 50blog tras5o9625699C56207"); user" 3 WSer LOUW, Wage s 10 ) 
{ Y iW s Object Iic (YS50blograo96256996560206Y); VISELW 9 Wuser OV, Vage s 9 ) 
{ W TeW s Object iIic(YS5S0blograso9625699656205Y); Vaser s WLS Bw, Vage 3 BJ 
EE EE EE EE s WISSE TW, Yaget s 7 } 
{ 4 ic s Object ic (Y50blobsrano96256996562034) D s Vusesr 6Y, Vage s © ) 
{ W ic a OoJecctiic(*¥50blbsraso96256996e562024)  Vuser™ s Wuser 5%, Yaga >s 5 ) 
{Wie EE EE EE e WISSP 4Y, Vage a 4 ) 
{ WW KE EE sS0blobrado9s2 569905602004); Vussert g Vuser SS, Vage 3 3 ) 
{ w eW g OgJjecciicl( Y S0olog raog 0250996C5602CETY) E 1 
{Ww We Ee EE EE z Wuser LW, Wage gz 1L 
> 


15.5 ”性 能 优化 


与 天 系 型 数据 库 一 样 ，MongoDB 虽然 性 能 早 越 ， 但 只 有 合理 及 有 效 的 优化 才能 使 数据 
库 性 能 得 到 最 大 程度 的 发 挥 。 在 MongoDB 中 ， 常 用 的 数据 库 优化 手段 包括 使 用 索引 、 固 害 
集合 、 使 用 Profile 优化 费 以 及 使 用 分 布 式 部 署 等 。 接 下 来 将 首先 介绍 索引 技术 。 


15.5.1 使 用 索引 


使 用 索引 能 够 明显 提高 数据 得 询 效率 。MongoDB 提供 了 多 种 数据 索引 方式 ， 前 面 已 经 
介绍 过 系统 在 创建 新 文档 时 ， 将 会 为 文档 主键 创建 唯一 索引 ， 这 些 索 引信 息 将 被 存放 到 
system.indexes 文件 中 。MongoDB 允许 开 友 人 员 为 其 他 文档 字段 或 属性 创建 索引 ， 创 建 格式 
如 下 。 

db.<table name>.ensureIndex({ 字 段 : 1}, {索引 类 型 : true}) 

其 中 索引 类 型 是 可 选 的 参数 ， 如 果 为 至 则 创建 单列 索引 。 可 选 的 值 有 unique 〈 唯 一 索 
引 ) 和 sparse〈 松 散 索 引 )。 有 款 引 字段 后 的 值 表示 索引 排序 ， 数 值 为 1 时 表示 升序 ，-1 表示 
倒序 。 接 下 来 将 结合 示例 代码 演示 索引 的 创建 与 管理 。 
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1. 创建 索引 
假设 数据 集 cl 有 10 条 数据 记录 ， 如 以 下 代码 所 示 。 


Slo SIL s ELNE y 


> 

{ w OW Ee EE EE Vaser s Wuser LW, Wagew g L et 
{ w iew 2 OoJeécrc EE EE p VuserY 9 Wuser 2w: GE g 2 } 
{ W i0 EE Vuser" s WSS Sp Vage s 3 ) 
{ W ic Ee EE EE Vuser" s Wuser AW, Wage s 4 ) 
{ W ic g Objecte (Y50blobsraio96256996562024) EE > 5 ) 
{ W ic 8s ObJectiicd(¥50blobgsraso9625699e56203Y); Vuser" s Wuser OO, Yaga s 6 ) 
{ W 6 s Object ic (Y 50blog eeh Ee E D E a Viser TY, Yaga s 7 ) 
{ W ict EE EE s Wuser 9Y- Vaoe 3 8) 
{ W _ ic 3 Object ric“ EE VuserY 3 Vuser 9Y, Wage 3 9 ) 
{ W ict 5 Object iic" 50blog tra5o9625699656207"); user" s WUSer LOW, Wage gs 10 ) 
> 


为 了 便于 演示 ， 这 里 将 使 用 age 字段 作为 条 件 查询 ， 并 且 使 用 exp 方法 观察 Bson 执行 
过 程 ， 如 以 下 代码 所 示 。 

> l. eL, EE ee 

{ 


EE 3 WBEASLECCUTSOLY, 
"nscanned" : 10, 
"nscannedObjects" : 10, 
w e 

Ne Ee 
EE 
"nChunkSkips™ : 0, 
"isMultiKey" : false, 
"jndexOnly™" : false, 
"indexBounds" : I 


} 


如 上 述 标 粗 代码 所 示 ，nscanned 表示 答 询 受 影 啊 行 数 《〈 受 影响 行 数 直 接 决 定 了 和 奉 询 效 
率 ); indexBounds 表示 得 询 所 使 用 到 的 索引 。 由 于 age RAUR, MUSEERNE. 
接 下 来 将 对 age 字段 创建 普通 索引 。 

> db.c1.ensureIndex({age:1}); 


索引 创建 完成 后 ， 接 着 再 次 查询 cl 数据 集 状态 ， 如 以 下 代码 所 示 。 


E ere Ee Dee Ree EE 
{ 


ee 
"nscanned" : 1, 
sccmeeloln ess 
E El 
KE e 
ee Ke 
A e SSS EDO 
"isMultiKey" : false, 
"indexOnly" : false, 
"indexBounds" : { 
steet ab 
[ 
1, 
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} 


通过 观察 执行 过 程 ， 可 以 看 到 创建 的 索引 已 经 被 应 用 到 了 age 字段 上 。 受 影响 行 数 也 由 
之 前 的 10 行 变 成 1 行 ， 这 就 意味 看 碍 询 引擎 由 全 表 扫 措 变 为 精确 扫 朱 ， 通 过 索引 后 但 询 效 
率 将 得 到 质 的 提升 。 

如 果 需 要 获取 当前 数据 集 学 段 罕 引 情况 ， 可 以 使 用 stats 或 getIndexKeys 操作 方法 进行 
但 看 ， 如 以 下 代码 所 示 。 


> db.c1.stats(); 
{ 


SW EE 

WerenoianeW 2 EIER 

WEE 

EE EE E 

EE ee BLIZ, 

EE EE El 

EE g Zy 

WilastmzcencoiLzeY RE 

"paddingFactor" : 1, 

"ares ER 

Ee E lee s 16352; 

Ve Be EE 
EE 
Jace IE > Sie 

} ， 

voku :| 


} 
> db.c1.getIndexKeys () ; 
PETAT M e EE | 


如 果 Bson 文档 字段 中 己 经 存在 大 量 数 据 ， 此 时 进行 索引 将 耗费 系统 大 量 资 源 。 
ensurcIndex 提供 了 1 个 参数 ， 用 于 将 操作 推送 到 后 人 台 。 如 以 下 代码 所 示 。 


> db.c1.ensureIndex({age:1} , {backgroud:true}) 


2. 删除 索引 

创建 索引 将 占用 大 量 的 磁盘 空间 ， 所 以 在 创建 索引 时 首先 需要 规划 好 Bson 字段 。 通 党 
情况 下 ， 只 有 用 于 条 件 人 查询 的 字段 才 需 要 创建 索引 ， 这 点 与 关系 型 数据 库 是 一 样 的 。 如 果 需 
要 删除 现 有 字段 索引 ， 可 以 使 用 dropIndex 操作 方法 ， 如 以 下 代码 所 示 。 


> db.c1.dropIndex ({age:1}); 
vobndexeswast E KK 


{ 
> db.c1.getIndexKeys () ， 
[ 
> 


a EIER 


如 果 需 要 清 衬 指定 数据 集中 的 所 有 索引 PARRA EER], H dropIndexes 操作 
方法 即 可 ， 如 以 下 代码 所 示 。 


> Ob. cl. dropindeszesil: 
d 
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e lee 3 2p 
"msg" : "non-_id indexes dropped for collection", 
er en EE 


ji 

> db.c1.getIndexKeys (); 
[ { WW Jel : 1 } ] 

> 


15.5.2 ”固定 集合 


固定 集合 是 MongoDB 中 一 项 重要 的 技术 ， 它 能 够 在 数据 可 控 的 范围 内 为 系统 提供 性 能 
出 色 的 储存 引擎 。 通 常情 况 下 ， 开 友人 员 创 建 的 数据 集合 无 论 是 容量 还 是 记录 条 数 ， 都 是 可 
变 的 ， 几 乎 没有 限制 。 但 固定 集合 将 改变 这 一 模式 ， 它 允许 开发 人 员 在 创建 集合 时 指定 容量 
或 记录 条 数 。 这 样 ， 当 数据 记录 达到 所 议定 的 临界 时 ， 回 定 集合 将 会 使 用 新 进入 的 数据 代 符 
先进 入 的 数据 〈 即 由 旧 到 新 做 老化 移出 处 理 )， 并 目 动 维护 数据 集合 对 象 顺 序 。 根 据 这 些 特 
性 ， 固 定 集合 非常 适合 用 于 日 六 处 理 、 消 县 队列 、 数 据 绥 存 等 应 用 。 接 下 来 将 深入 介绍 固定 
集合 的 使 用 。 

1. 创建 固定 集合 

在 前 面 的 内 容 中 ， 并 没有 涉及 集合 的 创建 。 事 实 上 集合 不 需要 手动 创建 ， 开 发 人 员 在 使 
用 find, inset 等 操作 方法 时 ， 系 统 会 日 动 做 集合 处 理 〈( 如 果 操 作 的 集合 存在 ， 则 直接 进入 
操作 流程 ， 否 则 首先 创建 相应 的 集合 再 执行 操作 流程 )。 当 然 ， 开 发 人 员 完 全 可 以 手动 显 式 
创建 集合 ， 如 以 下 代码 所 示 。 

> db.createCollection ("c11"); 

上 述 代 码 即 为 一 个 标准 的 集合 创建 方法 。 回 定 集合 的 创建 ， 在 方式 上 与 显 式 创建 集合 是 
一 致 的 ， 只 需要 声明 几 个 文档 关键 属性 即 可 。 格 式 如 下 。 

db .createCoolection ("和 集合 名 ”, {capped :true,size:100000,max:10}); 

如 上 述 代码 所 示 ，capped 属性 表示 集合 对 象 的 形式 ， 当 设置 为 true 时 表示 受 限 的 固定 集 
合 〈 默 认为 false); size 及 max 属性 只 有 当 capped 设置 为 mue 时 才 具 有 意义 。size 表示 集合 
容量 大 小 ， 以 学 节 为 单位 ，max 表示 集合 的 记录 受 限 数量 。 接 下 来 将 结合 示例 代码 ， 演 示 固 
定 集合 的 创建 ， 代 码 如 下 。 


> db.createCollection("logs", {capped:true,size:1000,max:10}); 
{ "ok" : 1 } 


上 述 代 人 码 表示 创建 一 个 名 称 logs 的 固定 集合 ， 并 设置 最 大 容量 为 1000B， 最 大 数据 记录 
为 10 &. size 和 max 属性 允许 同时 设置 ， 但 只 会 执行 其 中 一 个 属性 (由 受 限 条 件 先后 顺序 
决定 )。 接 下 来 将 问 该 固定 集合 插入 数据 ， 方 便 接 下 来 的 测试 。 


EE ni 


leet Ee ere VLOGS VLI)? 
Ee 
e EE E? 


Ee EE 
EECH 
[| 
EE Ee 
EE E 
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NM 一 一 一 一 一 


前 面 插 入 了 10 条 


Ven Ee E e 


MERO A e 


ebe PA 


Ek Sc e 


WO D 


6; EE 

EBO Ee 
KE SE 

ee 
EE g 


十 GGS 9， 
EE E 
EE 
SE 


Lol ee Lee a 


ulogs LOW } 
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志 记 录 。 由 于 在 创建 logs 固定 集合 时 ， 已 经 声明 了 最 多 可 接受 10 条 


记录 ， 所 以 以 下 的 操作 将 会 对 数据 进行 age-out《〈 老 化 移出 ) 处 理 ， 如 以 下 代码 所 示 。 


> for (i=11;i<16;i++){ 


V 一 一 一 一 一 一 一 一 一 一 V e 
e 
D 


> Clo logs. inssre (1 Les l; bocey g Logs VPLI) g 


ks 


vray 
AO he 


TETAN 


ebe PA 
VEN 
EE A 
TEAN 
aS A 


TEKAN 


elos LOGS s ELME) g 


METON e 


2 67 "body" : W Llogs_ 6" } 
SE EE 
2 WDO : logs 8" } 
EE a e e 
EE 
EE Kee) 
ee 
SE e Eeer 
EE 
EE 


可 以 看 到 ， 通 过 搬入 5 条 新 数据 ， 固 定 集 合 将 旧 数 据 (_id 为 1 一 的 数据 ) 移出 ， 并 在 


_id 为 10 的 数据 记录 后 添加 5 条 新 记录 (id 为 11 一 1 )。 


2. 普通 集合 转 固定 集合 


如 条 需要 将 现 有 的 普通 集合 转换 为 固定 集合 ， 可 以 运行 convertToCapped 命令 。 在 
MongoDB 管理 终 痕 中 ， 如 条 需 要 运行 额外 的 系统 命令 ， 可 以 使 用 runCommand 方法 ， 代 但 
如 下 所 不 。 


> db.runCommand({convertToCapped:"c5",size:10000});，; 


{ 


UU ok UU 


Ee 


上 述 操 作 表 示 将 c5 普通 数据 集合 转换 固定 集合 ， 并 有 限制 容量 为 10000B 。 这 里 需要 注 


意 的 是 ， 


看 即 可 ， 如 以 下 代码 所 示 。 


> ClO E ME E 


{ 


TASau e MESSE ech 
Teont o k 2 
e IOA, 


£ 


E EE EE 


"storageSize" 


3 12288; 


UTUME XESMIESME 3 ly 


EE EE 
"lasrtbastentäizen 
"paddingFactor" 
umoleke nno; 
"totalIndexSize" 


"indexSizes" : { 


3 12200, 


Set) 


Set 


max 属性 不 能 运用 于 convertToCapped 命令 。 转 换 成 功 后 ， 使 用 stats 操作 方法 得 
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} ， 
"capped" : 1, 
“maxu: 2 E EE 
vokael 

} 


如 上 述 标 粗 代码 所 示 ，capped 为 1 时 表示 该 集合 是 回 定 集合 。 固 定 集合 本 质 上 用 于 轮换 


数据 ， 所 以 并 不 需要 创建 和 索引。 执行 convertToCapped 命令 后 ， 原 有 的 字段 索引 也 将 会 被 删 
除 〈 包 括 主键 索引 )。 


15.5.3 GridFS 


MongoDB 是 一 个 文档 型 的 数据 库 ， 包 括 前 面 介绍 过 的 BSON 数据 ， 都 会 作为 文档 存 
储 。MongoDB 对 单 篇 文档 最 大 限 值 为 16MB。 这 就 意味 着 要 储存 大 型 文件 (例如 高 清 图 
片 、 视 频 、 压 缩 包 等 )， 使 用 Bson 数据 存储 系统 显然 是 不 能 满足 要 求 的 。GridFS 是 一 套 专 门 
用 于 存放 大 型 数据 的 MongoDB 储存 子 系统 。 利 用 GridFS 可 以 轻松 地 实现 大 文件 高 效 管理 ， 
例如 分 布 式 储存 、 故 障 转移 、 实 时 系统 热 扩 展 每 。 接 下 来 将 结合 示例 代码 ， 演 示 GridFS 的 
实战 应 用 。 

GridFS 的 管理 终 六 为 mongodbfiles， 在 使 用 方式 上 非 音 简单。 假设 需要 将 /datal/nginx- 
1.3.8.tar.gz 文件 存放 到 GridFS 系统 中 ， 只 需要 执行 如 下 命令 。 


[root@~]cd /usr/local/mongodb/bin 


[root@bogon bin]# ./mongofiles put /datal/nginx-1.3.8.tar.gz 

connected to: 127.0.0.1 

added file: { _id: ObjectId('50b323e0742ab0ded6ee713d'), filename: "/datal/nginx- 
1.3.8.tar.gz", chunkSize: 262144, uploadDate: new Date (1353917409646), md5: "538dce 
8d18b7a2c855134668q6078252", length: 738216 } 

done! 

mongofiles 终 冰 利用 的 有 3 个 命令 ， 分 别 是 put 〈 上 传 文件 )、，get〈 下 载 文件 )、1list《〈 列 
出 文件 )。 假 设 需要 奉 询 GridFS 中 的 文件 列表 ， 只 需要 运行 以 下 命令 即 可 。 


[root@bogon bin]# ./mongofiles list 


connected to: 127.0.0.1 

/datal/nginx-1.3.8.tar.gz (EIERE 

[root@bogon bin)? 

在 实际 应 用 开发 中 ， 通 各 需要 配合 crontab 或 者 inotify 使 用 (将 PHP 上 传 的 文件 定时 或 
实时 上 传 到 GridFS 系统 )。 


15.5.4 Profile 优化 器 


在 MySQL 中 可 以 通过 开启 慢 查 询 日 志 ， 从 而 得 出 最 需要 进行 优化 的 SQL 语句 。 在 
MongoDB 中 ， 同 样 提供 了 慢 查 询 日 记功 能 ， 在 使 用 方式 上 比 MySQL 更 加 简单， 下 面 分 


IINA. 
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1. 开局 慢 碍 询 日 志 

开局 MongoDB 慢 查 询 日 志 的 方式 共有 两 种 一 种 是 加 入 - profile 局 动 选 项 :; 另 一 种 是 
EEA imh EH setProfilingLevel 方法 设置 慢 租 询 日 六 级 别 。 下 面 分 别 介绍 。 

(1) -profile 局 动 选 项 

在 介绍 profile 启动 选项 前 ， 首 先 来 理解 profile 记录 级 别 。profile 共 支 持 3 种 级 别 ， 分 别 
DIF, 

> 0: 不 开局 。 

> 1: 记录 慢 命 令 (默认 为 >100ms)。 

Si: WURMAN Fe 

一 般 情 况 下 ， 只 需要 设置 为 1 即 可 。 

[root@bogon bin]# ./mongod --dbpath=/usr/local/mongodb/data/ --logpath=/usr/lol 


/mongodb/logs/mongodb.log --profile=1 --fork 

forked process: 2226 

(2) 使 用 setProfilingLevel 

如 果 已 经 局 动 MongoDB， 可 以 直接 使 用 setProfilingLevel 方法 设置 日 志 的 级 别 。 使 用 格 
式 如 下 。 

setProfilingLevel( level , slowms ) 

参数 level 表示 日 志 的 级 别 ， 文 持 的 值 与 profile 启动 选项 一 致 ， 参数 slowms 表示 慢 查 询 
超时 时 间 以 微 秒 为 单位 )。 如 以 下 代码 所 不 。 

3 


上 述 代码 表示 记录 所 有 超过 10ms 的 BSON 操作 。 

2. 查看 慢 查 询 日 志 

在 MongoDB 中 ， 所 有 日 志 都 被 存放 于 db 库 profile 数据 集 里 。 所 以 只 需要 查询 profile 
集合 即 可 获取 所 需要 的 日 志 信 息 。 例 如 查询 记录 时 间 为 10ms 的 记录 ， 代 人 码 如 下 。 


clo System. ororile rimal 4 millis 3 4 Seo» 10 ~ ) 


系统 还 提供 了 show profile 快捷 命令 ， 人 快速 租 询 最 新 5 条 超过 Ims 的 日 志 记 录 。 


15.6 在 MVC 中 使 用 MongoDB 


前 面 介 绍 的 内 容 都 是 基于 mongo shel 的 ， 接 下 来 将 利用 MongoDB 提供 的 PHP 扩展 实 
现在 PHP 中 直接 操作 MongoDB 数据 。 默 认 的 PHP 并 没有 提供 MongoDB 扩展 ， 需 要 开发 人 


AA SE 
员 上 自行 安装 。 


15.6.1 SS PHP 扩展 


MongoDB 的 扩展 安装 平台 主要 分 为 Windows 及 Linuxe Windows 下 的 PHP 扩展 安装 非 
种 简单， 读者 可 以 在 http://beauty-soft.net/book/php_mvc/down/php_monso_extension.html F $R 
到 相应 的 php_mongo.dll 扩展 文件 。 由 于 篇 幅 所 限 这 里 只 介绍 Linux 环境 下 的 安装 。 读 者 可 


以 在 http://pecl.php.net/package/mongo 中 找到 相应 的 PHP 扩展 包 ， 这 里 将 使 用 mongo-1.3.0 
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版 本 。 


[root@bogon] cd /datal 

[root@bogon datal]# wget http://pecl.php.net/get/mongo-1.3.0.tgz 
下 载 完 成 后 ， 解 压 即 可 。 

[root@bogon datal]# tar zxvf mongo-1.3.0.tgz 

[root@bogon datal]# cd mongo-1.3.0 

接 下 来 就 可 以 使 用 phpize 工具 创建 configure 文件 了 。 

[root@bogon mongo-1.3.0]# /usr/local/php/bin/phpize 


Configuring for: 


PHP Api Version: 20090626 

zend Module Api No: 20090626 

zend Extension Api No: 220090626 

[root@bogon  mongo-1.3.0]# ./configure --with-php-config=/usr/local/php/bin/php- 


config --enable-mongo 
creating libtool 
appending configuration tag “CXX” to libtool 
configure: creating ./config.status 
config.status: creating config.h 
configure MFH FEZA RERI, ol DL. bein 选项 得 看 所 文 持 的 配置 项 。 执 
行 完 configure 后 ， 还 需要 手动 进行 安装 。 过 程 如 下 。 


[root@bogon mongo-1.3.0]# make && make install 


Libraries have been installed in: 


/datal/mongo-1.3.0/modules 


If you ever happen to want to link against installed libraries 
in a given directory, LIBDIR, you must either use libtool, and 
specify the full pathname of the library, or use the `-LLIBDIR' 
flag during linking and do at least one of the following: 


add LIBDIR to the 'LD_LIBRARY_PATH' environment variable 


during execution 


add LIBDIR to the 'LD_RUN_PATH' environment variable 


during linking 
— use the '-Wl,--rpath -W1l,LIBDIR' linker flag 


— have your system administrator add LIBDIR to '/etc/ld.so.conf' 


See any operating system documentation about shared libraries for 


more information, such as the ld(1) and ld.so(8) manual pages. 
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Installing shared extensions: /usr/local/php/lib/php/extensions/no-debug-non-zts- 
20090626/ 

[root@bogon mongo-1.3.0]# ls  /usr/local/php/lib/php/extensions/no-debug-non-zts- 
20090626/ 


mongo.so sphinx.so 


出 现 上 述 信息 提示 ， 表 示 mongo for php Y REZEKI. SS FKF IF php.ini 文件 ， 并 加 
入 mongo 扩展 模块 引用 。 


[root@bogon mongo-1.3.0]# vi /usr/local/php/etc/php.ini 


;extension=php_shmop .dl1l 

extension=sphinx.so 

extension=mongo.so 

最 后 只 需要 重 司 php-fpm 即 可 。 

[root@bogon mongo-1.3.0]# pkill php-fpm 

[root@bogon mongo-1.3.0]# /usr/local/php/sbin/php-fpm 

SJE, mongo for php H ER ZEER J o M phpinfo 可 以 查看 到 mongo 选项 ， 如 
图 15-7 所 示 。 


图 15-7 php mongo 选项 


15.6.2 ”开启 MongoDB 用 户 验 证 


在 前 面 的 内 容 中 ， 细 心 的 谈 者 已 经 发 现 ， 我 们 在 登录 MongoDB 时 并 不 需要 进行 用 户 验 
证 。 事 实 上 MongoDB 已 经 提供 了 类 似 于 MySQL 的 用 户 权 限 验 证 ， 在 使 用 第 三 方 API Chal 
Wi PHP, Java) 调用 数据 库 时 首先 需要 开局 用 户 验证 。 步 又 如 下 。 

1. 开局 验证 

默认 情况 下 ，MongoDB 是 处 于 非 用 户 验 证 局 动 的 ， 这 种 模式 只 适用 于 快速 测试 数据 
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库 ， 在 生产 环境 使 用 时 还 需要 开局 用 户 权限 验证 。 开 局 用 户 权 限 验 证 只 需要 在 局 动 mongod 
时 加 入 --auth 选项 即 可 ， 如 以 下 代码 所 示 。 

[root@bogon bin]# ./mongod --dbpath=/usr/local/mongodb/data/ --logpath=/usr/local/ 
mongodb/logs/mongodb.log --auth -fork 

成 功 局 动 后 ， 此 时 MongoDB ET HI RI uER A ", HREH mongo 管理 工具 登 
录 即 可 ， 如 以 下 代码 所 示 。 


[root@bogon bin]# ./mongo 


MongoDB shell version: 2.0.8 

Copnnecting to: test 

这 里 需要 注意 的 是 ，MongoDB 使 用 db.admin 数据 集 存 放 超 级 管理 员 登 录 数 据 。 默 认 情 
况 下 admin 集合 数据 为 室 ， 所 以 束 算 开局 了 用 户 权限 验证 ， 系 统 依旧 不 会 对 空 用 户 进 行 任何 
WE 

2. 创建 用 户 

只 有 存在 用 户 数 据 ， 系 统 才 能 实现 权限 验证 。 这 里 首先 在 db.admin 数据 集中 创建 超级 
管理 员 。 过 程 如 下 。 

E 


EE ee 
{ 


EE 
"readOnly" : false, 
"owd" : "fa2121c224a64e8fd0f4f8c23999f7f5" 


} 

这 样 ， 用 户 root EHS SO Wir OI S.o dch, 2 imi e eao H ER E 
理 员 连接 数据 库 ， 而 上 只 需要 使 用 一 个 普通 管理 员 账 号 即 可 。 

MongoDB 人 允许 针对 特定 的 数据 库 创 建 管 理 员 ， 接 下 来 将 创建 一 个 “user1” 用 户 ， 议 用 
户 只 对 test 数据 库 有 管理 及 操作 权限 。 代 人 码 如 下 。 


E E 
switched to db test 
> db.addUser ("user1","123"); 


EE lg E EE EAN ke A E NUN Ok e an 
{ 

usert SS 

"readOnly" : false, 

"owd" : "fa2121c224a64e8fd0f4f8c23999f7f£5", 


w ICW > @oJjectradi" s0osedre72L1L367030a643526™) 
| 


这 样 ， 当 前 MongoDB 数据 库 系 统 中 束 存 在 2 个 用 户 了 。root 为 超级 管理 员 ; userl 为 
test 数据 库 的 管理 员 。 在 PHP 连接 test 数据 库 时 ， 只 需要 使 用 userl 管理 员 即 可 。 要 验证 前 
面 创建 的 用 户 ， 可 以 在 进入 mongo shell 前 进行 验证 ， 也 可 以 在 进入 mongo shell 后 使 用 
db.auth 方法 验证 。 


[root@bogon bin]# ./mongo -u userl -p123 


MongoDB shell version: 2.0.8 
connecting to: test 
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如 朱 验 证 失败 ，MongoDB 将 终止 操作 ， 如 以 下 代 但 所 示 。 


[root@bogon bin]# ./mongo 

MongoDB shell version: 2.0.8 

connecting to: test 

> show tables; 

Tue Nov 27 05:49:48 uncaught exception: error: { 
WE g Wunmnauchorizec! closcest lock EyYoe— 1 clients 127-00, 1W, 
“Eo s 10057 

i 


Es 


3. 在 PHP 中 连接 MongoDB 
通过 前 面 的 步 台 ， 接 下 来 就 可 以 在 PHP 中 连接 MongoDB 数据 库 ， 并 读 取 数据 库 中 的 数 
据 了 ， 如 以 下 代码 所 示 。 


<?php 
/ /连接 mongodb 
Sm = new Mongo ("mongodb://user1:123@192.168.2.16:27017/test"); 
// 选 择 数据 库 
$sdb = $m->test; 
// 选 择 数据 集合 
scollection = Ş$db->c8; 
Şarr=array ("age"=>array ('$gt'=>10)); 
// 调 用 find 方法 
SCUrsor = S$collesction=>finaq(Sacr) ， 
// 得 出 结果 
foreach ($cursor as $obj) { 
echo Sal rct] . Wor />% 
} 
// 关闭 链接 
$m->close(); 
?> 


如 上 述 代 人 码 所 示 ， 在 PHP 中 所 有 查询 语句 由 Bson 变 成 了 数组 。 事 实 上 只 要 掌握 Bson， 
使 用 PHP 数组 时 将 变 得 直观 、 人 简单 ， 因 为 这 两 者 在 PHP 中 是 可 以 互 换 的 。 


15.6.3 ThinkPHP 操作 MongoDB 


ThinkPHP 3.x 提供 了 对 MongoDB 的 全 面 文 持 ， 将 音 用 的 CURD 操作 进行 了 封装 ， 证 其 
更 适合 MVC 开发 。ThinkPHP 的 MongoDB 操作 模型 为 MongoModel， 用 户 设计 的 目 定 义 模 
型 需要 继承 于 该 模型 ， 才 能 进行 MongoDB 开发 。 下 面 将 分 别 介 绍 。 

1. 连接 数据 库 

默认 情况 下 ，ThinkPHP 只 提供 MySQL 驱动 ， 这 里 所 需要 的 MongoDB 数据 库 驱 动 需 要 
读者 目 行 前 往 官方 网 站 下 载 。 将 下 载 后 得 到 的 DbMongo.class.php 文件 复制 到 
ThinkPHP/Lib/Driver/Db/ 目 录 ， 这 样 就 完成 了 MongoDB 驱动 的 安装 。 然 后 就 可 以 配置 数据 
FERI, MERIIN MySQL 数据 库 配 置 相 类 似 ， 如 以 下 代码 所 示 。 


<?php 


Şarrl=include ("./config.inc.php");}; 
Šarr2=array ( 


'DB_TYPE' => 'Mongo', // 数据 库 类 型 
DB HOST"! 三 
'DB_NAME' => 'test', // 数据 库 名 
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'DB_USER' => 'userl', // 用 户 和 名 
'DB_PWD' EE // 密码 
'DB_PREFIX' => II, // 数据 库 表 前 级 
(E 
EE 


return array_merge ($arrl, sarr2); 
?> 


在 实际 应 用 开发 中 ， 读 者 可 以 使 用 独立 的 配置 项 存放 MongoDB Wei, In d 
态 切 换 即 可 。 这 里 为 了 方便 汗 示 ， 所 以 使 用 了 复 兰 方式 。 配 置 完 成 后 ， 要 测试 数据 库 是 合 连 
接 正 第 ， 最 直接 的 办 法 是 在 控制 占 动 作 中 输出 数据 集 信息 。 如 以 下 代码 所 示 。 


public function index (){ 


sm=new MongoModel('c8'); 
dump ($m) ; 
} 
2. CURD 操作 
接 下 来 将 结合 示例 ， 介 绍 ThinkPHP 对 MongoDB 常见 操作 的 支持 ， 分 别 为 查询 数据 、 
增加 数据 、 删 除数 据 、 修 改 数据 。 
(1) 查询 数据 
在 ThinkPHP 中 ， 开 发 人 员 可 以 使 用 高 效 、 易 用 的 连贯 操作 实现 MongoDB 的 CURD 操 
作 ， 并 且 在 操作 方式 上 与 其 他 数据 库 无 异 。 如 以 下 代码 所 示 。 
/** 
* AWA 
EH 


public function index (){ 


sm=new MongoModel('c8');}; 
ScearalY eY sarrtay (Virat; SOWY y 
Şrows=$m->where ($data)->limit (0,20)->order ("age desc")->select(); 
Şthis->assign("list",Ş$rows); 
Sthis=>cisplay () ; 
} 
可 以 看 到 ，ThinkPHP 中 的 连贯 操作 方法 ， 如 limit, order 等 都 可 以 直接 使 用 ， 系 统 
会 目 动 转换 为 MongoDB 所 支持 的 Bson 格式 。 对 应 的 视图 模板 index.html 文件 如 以 下 代 


但 所 示 。 


<ul elasss BC e 


GE Ee unt liset" E EA 


Ie 
a a a R < a a a 
</volist></ul> 


这 里 使 用 了 “lte” 表 达 式 ，ThinkPHP 与 MongoDB 之 间 的 查询 表达 式 如 表 15-4 PTR. 


SS 15-4 ThinkPHP 表达 式 与 Bson 表达 式 


ThinkPHP 表达 式 全 Bson 表达 式 
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ThinkPHP 表达 式 
gte ~ egt 
like 
mod 
in 
nin 或 者 not in 
all 
between 
not between 
exists 
size 
type 
regex 


exp 


(2) 删除 数据 


大 二 二 于 

模糊 查询 

取 模 运算 

in 查询 

not in 查询 
满足 所 有 条 件 
在 某 个 区 间 

不 在 某 个 区 间 
字段 是 否 存在 

限制 属性 大 小 

限制 字段 类 型 
MongoRegex 正则 查询 
使 用 MongoCode 查询 


WË 
xX 
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Bson 表达 式 


$exists 
$size 
$type 
$regex 


无 


单 击 “删除 ”链接 ， 将 进入 delete 控制 右 动 作 ， 在 该 动作 中 需要 完成 效 据 的 删除 操作 。 


代码 如 下 所 示 。 
/** 
* 删除 数据 


= Enter description bere ooo 


SC 


publie fumetion celete() i 


sm=new MongoModel('c8');}; 
if ($m->where (array ("_id"=>trim($_GET["id"])))->delete()) { 
$this->success ("删除 成 功 ")，; 


}else{ 


$this->error ("HIRR K"); 


} 


} 
(3) 修改 数据 


单 击 “ 修 改 ” 链 接 ， 将 进入 save 控制 磊 动 作 ， 在 该 动作 中 需要 完成 数据 修改 操作 。 代 


Wu Fra, 
/** 
* 修改 数据 


= macer description bere cooo 


人 


public function save (){ 


LE (emocy (S CaTt[Yict])) i 
$Sthis->error ("参数 错误 ")， 


} 


SS Dee EE Eegen Es 


if (!empty(S_POST) ) { 


sm=new MongoModel('c8'!'); 


$data["username"]=$_POST["username"]; 
Şdata["age"]=intval($_POST["age"]); 
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if (Sm->where (array ("_id"=>$pid))->save ($data))tft 
Sthis->success ("数据 修改 成 功 "); 
}else{ 


Sthis->error ("数据 修改 失败 ")， 

} 

}else{ 
sm=new MongoModel('c8'!'); 
Şrows=$m->where (array ("_id"=>$pid))->find(); 
Şthis->assign("data", $rows); 
Shs e SE EE 

} 


} 
对 应 的 视图 模板 save.html 文件 如 以 下 代码 所 示 。 


<form action=" URL /add/id/<!--{Ş$Think.get.id}-->" method="post" name="form1"> 
R ee a a Ee E 
=> >L 

< ea e pae ee e ee VI nae age le VN e daca el Aaa 
<p> 

<label> 

<input Ev ev nn nem = on id on value TENI Pi 

</label> 
</p> 
</form> 


(4) 增加 数据 

通过 前 面 的 和 学习， 增加 数据 同 梓 也 将 变 得 人 简单。 与 修改 数据 相 比 ， 只 需要 改变 操作 方法 
及 运行 迪 辑 即 可 。 如 以 下 代码 所 示 。 

/** 

* 增加 数据 

x Konter description /Mere ooo 

SH 

public function add(){ 

if (!empty($_ POST))I{ 
sm=new MongoModel('c8'"'); 


Şrows=$m->limit (1)->order ("age desc")—>select EI 
Şpid=array_keys (Şrows); 
$data["username"]=$_POST["username"]; 
şdata["age"]=intval (Ş$_POST["age"]); 
if ($m->add ($data)){ 
$this->success ("数据 添加 成 功 ")， 
Lelsel 
$this->error ("数据 添加 失败 ") ; 
} 
}else{ 
SE 


对 应 的 add bm 视图 模板 如 以 下 代码 所 示 。 


<form action=" _URL /add" method="post" name="form1"> 
Tt epe EE Size To name username! -> 
Ee a a a a 
<p> 

<label> 


458 O B 


第 15 章 使 用 MongoDB 


人 
</label> 

</p> 

</form> 


可 以 看 到 ，MongoDB 的 CURD 操作 在 ThinkPHP 中 变 得 直观 与 简单 ， 读 者 完全 可 以 结 


合 前 面 章节 的 内 容 ， 在 自 定 义 模型 中 实现 CURD 操作 ， 这 样 就 能 够 实现 更 加 灵活 的 模型 定 
制 ( 例 如 定义 主键 id、 字 段 限制 等 )。 


15.7 小结 


本 章 深 入 介绍 了 MongoDB 开发 的 方方面面 ， 涉 及 的 内 容 都 是 基于 实战 的 ， 读 者 在 学 习 
时 务必 根据 章节 内 容 动手 实验 。 

本 章 首 先 介绍 了 非 关 系 型 数据 库 与 天 系 型 数据 库 的 区 别 ， 接 看 通过 示例 的 方式 详细 演示 
了 Windows 平台 及 Linux 平台 上 的 MongoDB 安装 。 

接 看 通过 人 简单 明了 的 示例 ， 对 最 党 用 的 CURD 操作 分 别 进行 了 人 介绍， 方便 读 者 迅速 上 
F; 为 了 更 进一步 理解 MongoDB， 本 章 还 对 MongoDB 所 支持 的 查询 表达 式 进 行 了 深入 的 
讲解 。 

通过 这 些 内 容 ， 可 以 看 到 MongoDB 是 一 个 语法 严谨 、 使 用 灵活 的 数据 库 。 本 章 最 后 介 
HY MongoDB 的 局 级 使 用 ， 包 括 索 引 、 固 定 集合 、GridFS 大 文件 储存 等 。 读 者 在 完全 掌握 
这 些 内 容 后 ， 使 用 PHP 操作 MongoDB 将 变 得 轻松 及 高 效 。 

非 关 系 型 数据 库 最 显著 的 特点 是 高 效 、 有 灵活 。 本 章 介 绍 的 MongoDB 不 仅 可 以 用 于 数据 
库 系 统 ， 还 可 作为 文件 管理 系统 ， 事 实 上 很 多 NoSQL 数据 库 也 并 非 只 限定 于 存放 文本 数据 
的 ， 这 点 与 关系 型 数据 库 不 同 。 由 于 NoSQL 的 标准 之 一 区 是 key-value 储存 ， 这 种 灵活 的 储 
存 方式 ， 使 得 NoSQL 功能 得 到 无 限 扩展 。 例 如 接 下 来 将 介绍 的 Redis, WI 22 em 
NoSQL. 
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内 容 提要 


Redis 是 一 套 性 能 非常 高 效 的 内 存 数 据 储 存 系统 。 对 于 习惯 使 用 Memache 的 读者 而 言 ， 
使 用 Redis 是 非常 简单 、 直 观 的 ， 因 为 这 两 者 在 数据 结构 上 存在 许多 相同 之 处 。 本 章 首 先 介 
绍 Key-Value 储存 系统 的 特点 ， 然 后 进一步 介绍 Redis 作为 Key-Value 储存 系统 的 优势 所 
在 。 最 后 通过 示例 ， 详 细 演 示 每 个 Redis 操作 命令 的 使 用 。 

在 MVC 中 集成 Redis， 是 实现 高 效 编程 的 重要 前 提 ， 国 外 主流 MVC 框架 都 提供 了 
Redis 编程 支持 ， 但 国内 的 MVC 框架 只 能 实现 简单 的 缓存 应 用 。 本 章 继 续 以 ThinkPHP 作为 
MVC 框架 ， 详 细 介 绍 通过 模型 扩展 ， 实 现 Redis 编程 支持 。 本 章 内 容 是 10.3 节 的 延伸 ， 读 

需要 在 完成 该 章节 学 习 的 基础 上 才能 对 本 章 进 行 学 习 。 


学 习 目 标 


了 解 Key-Value 储存 的 概念 。 

了 解 Redis 的 优 缺 点 。 

理解 Redis 的 五 大 数据 类 型 。 

掌握 Redis 对 各 数据 类 型 的 操作 命令 。 
掌握 Redis 的 优化 。 

掌握 在 PHP 和 MVC 中 结合 Redis 进行 编程 。 
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16.1 Redis 的 使 用 


Redis 是 一 僚 局 性 能 、 多 用 途 的 Key-Value 存储 系统 。 在 众多 场合 中 均 可 以 使 用 Redis 代 
答 传 统 的 数据 库 或 者 缓存 系统 。 本 和 章 将 进一步 介绍 Redis 的 实战 运用 。 


16.1.1 Redis 概述 


本 书 10.3 市 已 经 对 Redis 进行 了 初步 介绍 ， 并 使 用 Redis NI Memcahed 作为 强 
、 完 善 的 缓存 系统 。 接 下 来 将 介绍 Redis 对 NoSQL 的 文 持 ， 方 便 读 者 在 MongoDB 与 

Redis 中 进行 选择 。 

Redis 是 更 加 彻底 的 Key-Value 储存 系统 ， 它 没有 专门 的 查询 语言 ， 也 没有 明确 的 数 
据 类 型 。 一 个 字符 串 ， 可 以 代 巷 所 有 的 储存 类 型 ， 例 如 直接 使 用 String 类 型 存放 传统 的 
文本 、 代 人 但、 序列 等 ;也 可 以 直接 存放 数据 流 ， 例 如 图 乒 、 视 频 等 ， 并 且 没 有 数据 大 小 
的 限制 。 

虽然 没 数 据 类 型 限制 ， 但 为 了 方便 数据 管理 ，Redis 提供 了 多 种 数据 结构 类 型 ， 分 别 为 
string CEFR), list JJK), sets (和 集合) 或 者 是 ordered sets (采集 合 )。 所 有 的 数据 类 
型 都 文 持 push/pop, add/remove, WRZ imi KÆR, sets 集合 型 别 等 操作 ， 这 些 操作 都 是 
其 有 原子 性 的 ，Redis 还 文 持 各 种 不 同 的 排序 功能 。 

与 Memcache 一 样 ，Redis 的 储存 方式 是 基于 内 存 的 ， 所 有 的 数据 谈 写 都 在 内 存 中 完 
Wo {H Memcache 使 用 的 是 libevent 库 ， 而 Redis 则 原生 使 用 了 epoll 并 步 通 信和 模型， 所 以 在 
性 能 上 比 Memcache 更 加 优秀 。 同 时 ，Redis 还 提供 了 Virtual Memory 功能 ， 使 得 数据 能 够 
在 指定 的 间隔 时 间 内 保存 到 便 盘 《〈 由 后 台 目 动 完 成 )， 避 免 数 据 在 内 存 中 丢失 。 

与 MongoDB 一 样 ， 虽 然 同 称 为 NoSQL， 但 Redis 无 论 在 使 用 方式 上 ， 还 是 数据 处 理 流 
程 上 都 存在 较 大 区 别 ， 这 对 于 初学 者 而 言 将 会 造成 一 些 困 惑 。MySQL、Redis、MongoDB、 
Memache 这 三 者 之 间 的 属性 对 比如 表 16-1 ra, 


SS 16-1 储存 属性 概念 


与 MongoDB 相 比 ，Redis 主要 优点 分 别 如 下 。 

> Redis 数据 储存 在 内 存 中 完成 ， 所 以 在 速度 上 比较 具有 优势 ，MongoDB 使 用 的 
memory-mapped 处 理 方式 本 质 上 还 是 磁盘 操作 。 

> Redis 与 MongoDB 在 设计 之 初 均 考 虑 到 分 布 式 处 理 能 力 ， 所 以 这 两 者 都 提供 了 集群 
部 赦 配 置 接口 ， 但 相对 而 言 Redis 集群 部 署 更 加 容易 及 稳定 。 

> Redis 提供 了 简单 的 事务 文 持 ，MongoDB 不 文 持 事务 。 
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Redis 并 非 只 有 优点 ， 也 有 缺点 ， 这 些 缺 点 是 明显 的 ， 分 别 如 下 。 

> Redis 没有 字段 的 概念 ， 所 以 在 数据 查询 上 功能 比较 弱 ， 文 持 的 特性 比较 简单 。 

> Redis 单个 value 的 最 大 容量 可 达 1GB ， 虽 然 MongoDB 单个 文档 最 大 容量 为 

16MB， 但 MongoDB 提供 了 GridFS 用 于 实现 超大 文件 存储 。 
> 由 于 Redis 本 质 上 是 一 个 内 存 数 据 库 ， 上 所 以 内 存 使 件 的 容量 大 小 直接 决定 了 Redis 可 
用 的 数据 库 空间 (Redis 2.0 ri% Y Virtual Memory 功能 解决 了 容量 问题 ， 但 虚拟 内 
仓 本 质 上 束 是 磁盘)。 

> Redis 虽然 在 内 存 中 查询 数 据 ， 但 为 了 人 确保 数据 的 安全 ，Redis 默认 情况 下 使 用 子 线 
程 对 数据 进行 持久 化 处 理 ， 如 果 配 置 不 当 《〈 默 认 刷 新 间 隅 为 208)， 将 会 使 系统 运行 
效率 适得其反 。 

> MongoDB 提供 了 mapreduce 数据 分 析 功 能 ，Redis 则 没有 数据 分 析 功 能 。 

随 着 大 数据 的 爆发 ， 无 论 是 Redis 还 是 MongoDB， 都 将 会 迎 来 越 来 越 多 的 功能 更 新 ， 
选择 哪 一 个 作为 NoSQL 数据 库 ， 要 看 项 目 需要 及 开发 人 员 技 术 水 平 。 现 在 对 这 两 种 数据 库 
进行 总 结 及 对 比 还 为 时 过 早 。 而 Memcache 已 经 好 入 没 更 新 了 ， 上 所 以 无 论 从 性 能 、 功 能 还 是 
后 续 更 新 考虑 ，Redis 都 是 Memcache 的 理想 和 苞 代 品 ， 除 此 之 外 Reids 还 非常 适用 于 以 下 
场合 。 

(1) 海量 的 简单 读 写 

根据 Redis 官方 的 测试 结 来 ， 在 50 个 并 发 的 情况 下 请 求 10 万 次 ， 写 的 速度 是 110000 
次 /s， 读 的 速度 是 81000 次 /s。 上 所 以 用 于 处 理 类 似 于 微 博 热 点 事件 、 焦 点 排序 、 关 注 排 行 之 
类 实时 但 功能 单一 的 局 部 应 用 是 最 理想 的 。 

(2) 实时 的 反 垃 圾 系统 

肥 垃 圾 系统 通常 都 是 基于 关键 词 的， 使 用 Redis 储存 关系 词 ， 能 够 利用 Redis 的 高 性 
能 ， 为 监控 系统 提供 稳定 及 精确 的 实时 监 探 功能。 典型 的 应 用 有 邮件 系统 、 评 论 系统 等 。 

(3) 公开 的 统计 系统 

利用 Redis 的 简单 易 用 的 群体 部 团 功 能 ， 能 够 轻易 地 开发 大 规模 的 统计 系统 ， 例 如 广告 
系统 、 网 站 统计 系统 、 简 单 的 股市 分 析 系 统 等 。 

(4) 消息 队列 

消息 队列 是 提高 关系 型 数据 库 数据 插入 的 有 效 手段 ，Redis 是 NoSQL 中 为 数 不 多 文 持 消 
县 队列 的 数据 库 。 第 用 的 应 用 有 网 站 邮件 、 站 内 短信 、 评 论 系统 等 。 

(5) 消息 堆栈 系统 

消息 扒 栈 与 消息 队列 在 处 理 顺 序 上 是 相反 的 ， 消 县 队列 的 规则 为 先进 先 出 ;而 消 县 堆栈 
则 是 先进 后 出 。 典 型 的 应 用 有 论坛 回帖 排序 等 。 

(6) 日 志 或 缓存 系统 

Virtual Memory 功能 非常 适用 于 存放 安全 日 志 、 网 站 日 志 以 及 数据 缓存 等 。 国 内 的 狐 浪 
微 博 所 构建 的 缓存 系统 是 全 球 最 大 的 Redis RFRA. 


16.1.2 ”常用 管理 命令 


Redis 的 安装 方式 在 本 书 10.3.1 节 已 经 详细 介绍 过 ， 这 里 不 再 重 述 。Redis 安装 完成 后 ， 
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可 以 在 安装 路 径 bin 目录 中 找到 redis-cli 管理 终 
Redis 数据 库 。 命 令 运行 过 程 如 下 。 


[root@bogon ~]# cd /usr/local/redis/bin/ 
[root@lbogon bin]# ./redis-cli 
redis 127.0.0.1:6379> 

zk dn As 


进入 管理 终端 后 ， 直 接 输 入 key * 命 令 ， 
Memcached 中 的 key)， 如 以 下 代码 所 示 。 


Sods EE 


可 以 全 看 当前 Redis 


le) 
2) WkEYSW 
e GESIN 

E, SS 

el Ee Se 
G6» SE 
E Sy 
EE 
9) Eeaerk- 
10) "num" 
LL) waeanmew 
REESEN 
dee 
Al 


redis 127.0.0.1:6379> 
与 mongo 管理 终端 一 样 ，redis-cli 管理 终 
学 习 ， 这 里 首先 对 redis-cli 利用 的 数据 库 管 理 命令 进行 介绍 ， 
flushdb : 清空 数据 库 。 
select db-index: 通过 索引 选择 数据 库 ，0 表示 所 有 。 
检查 key ERTE o 
move: 移动 key。 
del: 删除 key. 
rename: 更 改 key 名 称 。 
expire: 设置 key 超时 。 
ttl key: 永久 化 剩余 时 间 〈-1 则 表示 已 经 完成 持久 化 )。 
persist: 立即 执行 key 数据 永久 化 。 
type: 得 到 key 数据 类 型 。 
randomkey: 随机 返回 key. 
save: 将 数据 同步 保存 到 人 磁盘。 
bgsave: 将 数据 异步 保存 到 人 厂 熏 。 
lastsave: 返回 上 次 成 功 将 数据 保存 到 磁盘 的 Unix 时 间 惟 。 
shundown: 将 数据 同步 保存 到 磁盘 ， 然 后 关闭 服务 。 
提供 服务 器 的 信息 和 统计 。 
monitor: 实时 转 储 接收 到 的 请 求 。 
slaveof: 改变 复制 策略 设置 。 
config: 在 运行 时 配置 Redis 服务 器 。 


exists: 


info: 


VW WW WW VV WW WW WWW WM WM WV 


io, =J MongoDB 一 样 ， 直 接 运 


SL 
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ZÍTRNPJ A 


数据 库 所 有 Key〔 类 似 于 


理 命 令 。 为 了 方便 后 向 的 


分 别 如 下 。 
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> quit: 关闭 连接 (connection). 

> auth: 简单 密码 认证 。 

Redis 虽然 称 为 数据 库 ， 但 实际 上 Redis 的 数据 库 与 MongoDB、MySQL 等 传统 数据 库 
存在 概念 上 的 区 别 。Redis 数据 库 是 预先 设计 的 ， 没 有 提供 创建 库 的 命令 ， 但 开 友 人员 可 以 
在 配置 文件 中 定义 Redis 数据 库 数 量 ， 默 认为 16 个 ， 如 以 下 代码 所 示 。 


# dbid is a number between 0 and 'databases!'-1 
databases 16 


默认 情况 下 ，redis-cli 将 定位 到 索引 写 为 0 的 数据 库 。 要 手动 定位 到 指定 的 数据 库 ， 可 
以 使 用 select 命令 ， 然 后 传 入 数据 库 索 引号 即 可 《从 0 开始 )。 如 以 下 代码 所 未 。 
SCILS 127-0:0-1:6379> Ek 


OK 
recie 127.0: 0. Ls 63V791L5]> 


FERNER KREA RI TAN 15 的 数据 库 ， 所 有 key RIE EKRI PAA E o 


16.1.3 Redis 用 户 验 证 


由 于 Memcache 不 提供 用 户 登 录 验 证 功能 ， 所 以 在 部 团 时 必须 将 其 内 置 于 内 部 网 络 中 ， 
这 对 分 布 式 部 普 将 会 造成 安全 性 问题 。 在 Redis 中 ， 只 需要 在 配置 文件 中 设置 requirepass 选 
项 ， 即 可 开局 人 窗 单 的 用 户 验 证 功能 ， 从 而 有 效 提 高 安全 性 ， 步 又 如 下 。 

首先 打开 redis.conf 配置 文件 ， 命 令 如 下 。 


[root@~]#cd /usr/local/redis/etc 


[root@bogon etc]# vi redis.conf 
在 打开 的 redis.conf 配置 文件 后 ， 找 到 requirepass foobared 配置 项 ， 复 制 并 粘贴 该 配 置 
数据 ， 如 以 下 代码 所 示 。 


# use a very strong password otherwise it will be very easy to break. 
# 


# requirepass foobared 


requirepass 123456 


如 上 述 标 粗 代码 所 示 ， 表 示 Redis 登录 密码 设置 为 123456。 完 成 后 ， 保 存 redis.conf Bi 
aL, HEIM redis 即 可 。 命 令 如 下 。 


[root@bogon etc]# pkill redis-server 


[root@bogon etc]# /usr/local/redis/bin/redis-server /usr/local/redis/etc/redis.conf 

通过 前 面 的 步骤 ，Redis 就 具备 用 户 验 证 功能 了 。 在 使 用 前 登录 用 户 必须 要 进行 授权 才 
能 进行 操作 ， 人 否则 将 出 现 权 限 铬 记 ， 如 以 下 代码 所 示 。 

recis 二 2 0 0 156379> keys * 


(error) ERR operation not permitted 
recis 127-0- 0. 1; 65379> 


与 MongoDB 一 样 ， 在 Redis 中 进行 用 户 授权 共有 两 种 方式 。 一 种 在 登录 redis-cli 管理 
终端 时 加 入 -a 授权 项 ， 并 传 入 授权 密码 即 可 ， 如 以 下 代码 所 示 。 

[root@bogon bin]# ./redis-cli -a 123456 

另 一 种 则 是 进入 redis-cli 管理 终端 后 使 用 auth 命令 授权 。 如 以 下 代码 所 示 。 

eol 27.0.0,.186379> EE 


OK 
EE ER E EE 
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由 于 Redis 是 以 key 作为 储存 单位 的 ， 所 以 用 户 验 证 显得 比较 简单 。 只 能 对 全 局 的 单 
用 户 进行 验证 ， 不 像 MongoDB 那样 支持 多 用 户 验 证 ， 这 点 在 多 人 管理 的 系统 中 ， 将 会 造 
成 不 便 。 


16.2 Redis 数据 类 型 


数据 类 型 是 Redis 数据 库 中 比较 重要 的 概念 ， 它 是 数据 结构 中 的 基础 储存 单元 。 在 
Redis 中 ， 数 据 类 型 直接 与 系统 的 功能 相 挂 钓 ， 例 如 如 果 要 做 一 个 队列 服务 器 ， 需 要 使 用 list 
数据 类 型 ， 而 如 果 只 和 需要 作为 绥 存 服务 器 ， 只 需要 使 用 String 数据 类 型 即 可 。Redis 共 文 持 5 
种 数据 类 型 ， 从 一 定 的 角度 讲 ，5 种 数据 类 型 代表 了 5 种 应 用 场合 。 接 下 来 将 深入 介绍 
Redis 所 文 持 的 数据 类 型 。 


16.2.1 String 类 型 


String 类 型 是 Redis 中 最 基础 的 数据 类 型 ， 它 使 用 典型 的 key-value 方式 存放 及 但 询 数 
据 ， 在 使 用 方式 上 与 Memcache 类 似 ， 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> set name ceiba 
OK 
EE L27- 0-071: 63792 


上 述 代 但 表示 设置 一 个 名 为 name 的 字符 串 类 型 key， 并 赋予 name 的 值 为 ceiba。 与 
Memcache 不 同 的 是 ， 在 Redis 中 所 有 字符 串 都 是 使 用 二 进 制 进行 保存 的 ， 从 而 确保 数据 
的 安全 与 高 效 。 当 然 ， 开 发 人 员 也 可 以 直接 存放 二 进 制 字符 流 ， 例 如 图 片 、 视 频 、 可 的 
TT VCF, 

如 条 要 获取 字符 串 类 型 的 key， 只 需要 使 用 get 命令 即 可 ， 如 以 下 代码 所 示 。 

redis 127.0.0.1:6379> get Dame 


"ceiba" 
creches L27-0. 0,1l: 631797 


set 及 get 是 字符 串 类 型 中 最 简单 的 命令 ， 相 信 读 者 能 够 轻易 和 掌握。 当然 ，Redis 还 提供 
了 其 他 非常 有 意义 的 字符 串 操 作 命 令 ， 下 面 将 分 别 介 绍 。 

(1) mset 

mset 命令 用 于 提高 管理 终端 的 输入 速度 ，mset 允许 开发 人 员 一 次 性 设置 或 更 改 多 个 字 
符 串 key。 如 以 下 代码 所 示 。 

redis 127.0.0.1:6379> mset namel ceibal name2 ceiba2 

OK 

redis 127.0.0.1:6379> get namel 

Eed 

Eeer Ee Ger nemez? 


"ceiba2" 
redis 127.0.0.1:6379> 


(2) msetnx 
与 mset 命令 相似 ，msetnx 也 是 用 于 设置 多 个 key 的 值 。 但 msetnx 是 原子 性 的 ， 所 以 不 
文 持 修 改 ， 运 行 时 要 确保 每 个 key 必须 都 是 不 存在 的 ， 并 且 都 是 可 设置 成 功 的 ， 束 算 存 在 其 
中 一 个 key 操作 失败 ， 整 条 命令 也 将 终止 执行 。 如 果 运 行 成 功 ， 将 返回 结果 1， 否则 返回 
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0。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> msetnx name3 ceiba3 name4 ceiba4 


(integer) 1 


edis 127.0.0.1:6379> msetnx name3 ceiba3 name4 ceiba4 


(integer) 0 
recitis L127,.060.1s 


(3) incr 


6379> 


如 果 key 的 值 为 数字 ， 
加 1， 如 以 下 代码 所 示 。 


EE 2 0: 
OK 

e Ee 
(integer) 31 
os L2 le Le 
(integer) 32 
Techie T2700 
EEN 

e e Ee 


(4) incrby 


6379> 


6379> 


6379> 


6379> 


6379> 


可 以 使 用 incr 命令 进行 加 运算 。 该 命令 运行 一 次 ，key 数值 将 被 


set age 30 
incr age 
incr age 


get age 


incrby 与 incr 命令 类 似 ， 不 同 之 处 在 于 incrby 会 目 动 创 建 相 应 的 key〈 当 指定 的 key 不 
存在 时 )。 并 且 incrby 可 以 指定 加 值 的 数值 ， 而 不 是 每 次 只 能 加 1。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> set ran 10 


OK 


redis 127.0.0.1:6379> incrby ran 5 


(integer) 15 
EE 


6379> 


如 果 指 定 的 key 不 存在 ，incrby 将 会 目 动 创建 。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> incrby rans 5 


(integer) 5 
e E Oe 


(5) append 


6379> 


如 末 需 要 在 指定 的 key HÆRE TIF, EH append 命令 。 该 命令 运行 成 功 后 


将 返回 新 key 值 的 长 度 。 如 以 下 代码 所 示 。 


EE Ee 
OK 

EE 127:-0-0; de 
"Welcome to the 
EOLS 1127-0-0, 1g 
(integer) 30 

e DE Ok 
"Welcome to the 
EC L27- 0:0 ig 


(6) getrange 


6379> 


6379> 
redis 
6379> 


6379> 
redis 
6379> 


set myname "Welcome to the redis " 


get myname 


UU 


append myname "thank you" 


get myname 
thank you" 


getrange 类 似 于 PHP 编程 中 的 substr 函数 ， 用 于 截取 指定 位 置 中 的 字符 串 。getrnage 命 
令 支 持 两 个 参数 ， 分 别 为 字符 串 的 开始 位 置 及 结束 位 置 。 假 设 需要 获取 字符 串 myname 中 的 


HU a Ter, TA Fro, 
red 27.0.0.186379> ger myname 


"Welcome to the redis thank you" 


redis 127.0.0.1:6379> getrange myname 0 5 
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"Welcom" 
Techie 1127-0-0. 1:6379% 


(7) mget 
mget 用 于 获取 多 个 key 值 ， 并 自动 按照 批量 得 询 有 顺序 进行 排序 。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> mget KI k2 k3 


1) W RW 
2) WD W 
3) u QOY 


redis 127.0.0.1: 63979> 


16.2.2 Hash 类 型 


String 类 型 只 能 储存 单一 的 键 值 对 ， 特 别 适 合 存放 数值 统计 。Hash 类 型 对 String 类 型 进 
行 了 水 平 扩展 ， 改 变 了 一 个 Value 只 能 存放 单条 数据 的 形式 ， 而 是 一 个 Value 对 应 一 个 Hash 
表 ， 如 图 16-1 所 示 。 


图 16-1 Hash 数据 类 型 


Hash 类型 是 以 虚拟 表 的 形式 存放 数据 的 ， 特 别 适 合 储存 对 象 。 理 论 上 将 一 个 对 象 存 放 
到 内 存 中 ， 相 较 于 使 用 String 类 型 更 加 高 效 。 接 下 来 将 介绍 Hash 类 型 的 常用 命令 。 

(1) hset 

hset 用 于 创建 Hash 数据 类 型 ， 格 式 如 下 。 

hset 键 字段 值 1 [ 值 2.…] 

如 有 条 创建 的 key 不 存在 ， 则 首先 创建 并 返回 1， 否 则 终止 执行 并 返回 0。 如 以 下 代 
公所 示 。 


redis 127.0.0.1:6379> beet h1 myfield book 

(integer) 1 

SHS 127:-0:0.-136379> naee ml myirelel Dook 

(integer) 0 

(2) hmset 

hmset 命令 用 于 提高 管理 终端 的 输入 速度 ，hmset 允许 开发 人 员 一 次 性 设置 或 更 改 多 个 
Hash 罕 段 值 。 如 以 下 代码 所 示 。 

redis 127.0.0.1:6379> hmset h2 fieldql v1 field v3 

OK 

EGR 127.0, 0, L: 6379> 

(3) hsetnx 
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设置 Hash 字段 值 为 指定 的 值 。 与 hset 不 同 ，hsetnx 个 文 持 修改 。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> hsetnx h3 myfield v1 
(integer) 1 

recs 127-0.0-1:6379> hnhgecnz N3 myrLiele V2 
(integer) 0 

SCHED L127.0.0.,136373> 

(4) hget 


获取 指定 Hash 字段 值 。 如 果 获 取 的 字段 不 存在 ， 则 返回 nil。 如 以 下 代码 所 未 。 
redis 127.0.0.1:6379> hget h3 myfield 

"yq" 

reeche 127,0. 0,136579> NGEL AS myi E ee 

(nil) 

于 SG 总 127,.0.0,136379> 


(5) hmget 

如 果 使 用 了 hmset 设置 了 多 个 Hash 字段 ， 可 以 使 用 hmget 命令 批量 获取 多 个 Hash F 
段 。 命 令 运 行 成 功 后 ，hmset 会 按照 指定 的 Hash 字段 进行 排序 。 如 以 下 代码 所 示 。 

redis 127.0.0.1:6379> hmset h5 name ceiba age 25 

OK 

redis 127.0.0.1:6379> hmget h5 name age 

W Cl a 

2) W 25W 

Gi binerbw 

(6) hincrby 

hincrby 命令 可 以 为 指定 的 Hash 数 人 字段 值 增加 指定 的 数值 。 命 令 运行 成 功 后 ， 将 返回 
修改 后 独 的 字段 数值 。 如 以 下 代码 所 示 。 

TGS 127:-0:-0-1:;6379> MSee ne age 20 

(integer) 1 

redis 127.0.0.1:6379> hincrby h6 age 10 

(integer) 30 

EOLS 127-0., 0,136379> 

(7) hexists 


检测 指定 的 Hash 字段 值 是 侍 存 在， 如 果 存 在 则 返回 1; FURE Oo, aA Fra, 
ECS 站 2 00. Ireiipz besxists n5 name 

(integer) 1 

redis 127.0.0.1:6379> hexists h5 names 

(integer) 0 

recs ll2750.0.1sG6379> 


(8) hlen 
返回 指定 Hash 字段 的 长 度 ， 如 果 指 定 的 字段 不 存在 则 直接 返 回 o. nl FRETZ. 


redis 127.0.0.1:6379> hlen h5 
(integer) 2 

EE 127-0,.0,.156379> hlen nL5 
(integer) 0 

echis 127-0: 0., 15 6319> 

(9) hdel 


删除 指定 的 Hash "RF, iech tier, WARIRI O0. eA FRIR o 
Lehes 0 0, 18 65379> bmoer M5 NEME age 

He E 

Bee 
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redis 127.0.0.1:6379> hdel h5 age 
(integer) 1 
recs 127.0.: 0. 1s 6379> 


(10) hkeys 

返回 指定 Hash Key 所 有 学 段 名 称 。 如 末 指 定 的 字段 不 存在 ， 则 返回 宇 值 。 如 以 下 代码 
所 示 。 

recs O00 60> hkeysen 

L) walue, 

2) "72" 

EE 1127-0-01 6379> 

(11) hvals 

返回 指定 Hash Key 所 有 字段 值 。 如 末 指 定 的 字段 不 存在 ， 则 返回 衬 值 。 如 以 下 代 
t PE ZR o 

cecis 2750.0. L6379> Mmeee MO riell vi rielo? y2 

OK 

redis 127.0.0.1:6379> hvals h9 

LY WEE 

2) fe 

O oe ea 

(12) hgetall 


返回 指定 Hash Key 所 有 字段 名 称 及 字段 值 。 如 果 指 定 的 学 段 不 存在 ， 则 返回 衬 值 。 如 


以 下 代码 所 示 。 
redis 127.0.0.1:6379> hgetall h9 
1) WELSlLelLY 
A E 
Sue E 
4) "v2" 


echie 1127-0-01: 6379% 


16.2.3 List 类 型 


Hash 类 型 是 一 个 哈 稀 表 结 构 ， 而 List 类 型 则 是 一 个 链表 结构 ， 所 以 非常 适合 储存 有 序 
( 正 序 或 倒序 的 队列 数据 。 开 发 人 员 可 以 通过 ljpush、]lpop 命令 对 链接 表 中 的 队列 数据 进行 
压 入 或 弹出 ，Redis 正 是 基于 list 链接 表 ， 实 现 列 队 或 堆栈 服务 的 。 如 图 16-2 所 示 。 


one 
two 
three 


four 


five 


SIX 


图 16-2 list 链接 表 
如 图 16-2 所 示 ，list 链接 表 结 构 中 的 元 素 (one 到 seven) 通常 情况 下 是 有 序 的 ， 也 可 
以 是 无 序 的 ， 元 素 名 称 可 以 相同 。 使 用 lpush 命令 对 元 素 由 上 到 下 顺序 进行 车 加 ， 命 令 运 行 
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成 功 后 将 返回 当前 列表 的 元 又 数量 ， 如 以 下 代码 所 未。 
redis 127.0.0.1:6379> lpush mylist one 
(integer) 1 


如 果 要 获取 指定 的 list 链接 表 元 素 ， 使 用 lrange 命令 即 可 。 该 命令 共有 两 个 参数 ， 分 别 
表示 列表 奋 询 范围 《从 上 到 下 由 0 开始 ，-1 表示 列表 最 后 一 个 元 素 )。 如 以 下 代 但 所 示 。 


人 GBS L127.0.0 136379> liramee mylige O =I 


1) "one" 
2) "one" 
3) "one" 


除了 lpush 及 lrange 命令 之 外 ，List 数据 类 型 操作 命令 常用 的 有 rpush, linsert, Iset 等 。 
下 面 将 结合 示例 代码 分 别 介绍 。 

(1) rpush 

rpush 与 lpush 命令 类 似 ， 不 同 之 处 在 于 rpush 的 添加 顺序 是 相反 的 ， 也 惑 是 说 rpush 由 
下 往 上 进行 元 聂 压 入 。 如 以 下 代码 所 示 。 


EON 127.060。1803)9> mous myliste one 


(integer) 3 
EE EE 


(2) linsert 

在 指定 的 list ERRIZ or ID Oe Tu ärt, op terre rd gu we re, 
如 以 下 代码 所 示 。 

redis 127.0.0.1:6379> linsert mylist1 before one 1 


(integerl 3 
EOL 1127:0013 60379> 


上 述 代 但 表示 在 mylistl HEZK “one” JERAN INJU “1”. SAE ol NEIER 
JCA ja HEMT FERRE, R m REM before 指令 为 after 命令 即 可 ， 如 以 下 代码 所 示 。 
redis 127.0.0.1:6379> linsert mylist1 after one 1 


(integer) 4 
recs 1127.0.0.1863795> Llrange mylistil 0 = 


1) "two" 
2) WG W 
3) "one" 
4) wW W 
(3) lset 


每 一 个 元 素 被 添加 进 list 链接 表 后 ， 都 将 得 到 一 个 唯一 的 索引 号 。 使 用 Jset 命令 ， 通 过 
唯一 的 索引 号 ， 就 可 以 实现 对 列表 的 修改 。 如 以 下 代码 所 示 。 


EE lranee mylistr © =I 


ki ewe) 

Zl mew 

21 Donen 

EE EE Ee myliste 2 name 
OK 

EE 127.0.0.186379> lranee mylisrtr 0 =I 
L) vone” 

2I omey 

3) "name" 

(4) lrem 


从 指定 的 list 连接 表 中 批量 删除 相同 的 元 素 。 删 除 格 式 如 下 。 
lrem 列表 key 删除 数量 同名 元 素 


470 0 E 


第 16 章 Redis 实战 


其 中 删除 数量 必须 是 一 个 数值 ， 可 以 是 正 数 或 负数 。 当 删除 数量 为 正 数 时 表示 由 上 到 下 
顺序 删除 ， 如 果 为 负数 时 则 由 下 往 上 进行 删除 ， 如 以 下 代码 所 示 。 


SCS 127.0.0.1386379> lramee mylisrtr 0 =i 


E 
2) vone“ 
S) Tonei 
4) "one" 
5) "one" 


6) "name" 

redis 127.0.0.1:6379> lrem mylist 4 one 
(integer) 4 

creches 127:-0:0-136379> Lrange mylisrtr 0 =I 
1) "one" 

2) "name" 

cecais 127-0.0.156379> 


(5) ltrim 

批量 删除 指定 的 list 链接 列表 元 素 ， 并 保留 指定 位 置 的 元 素 ， 使 用 格式 如 下 。 

ltrim 列表 key 保留 开始 位 置 保留 结束 位 置 

其 中 “保留 开始 位 置 ” 和 “保留 结束 位 置 ” 必 须 为 数值 类 型 (从 0 开始 ，-1 表示 最 后 
一 个 元 素 )， 下 和 面 将 通过 代码 演示 ltrim 命令 的 使 用 。 


CLS L127.0.0.1806379> Lrange mylistz 0 =I 


I ) aR AS 

E WAS fo 

21 Ythree" 

4) "two" 

5) "one" 

recs 127-0.-0-136379> Ikesi SEE 4 =i 
OK 

recis 127.-0-0.1:6379> Llrange mylist2 0 =1 
Tone" 

(6) lpop 


从 指定 的 list 链接 列表 中 由 上 到 下 单个 弹出 删除》 元 素 。 如 果 执 行 成 功 将 返回 被 弹出 
的 元 素 名 称 ， 人 否则 返回 nil。 如 以 下 代码 所 未 。 


EE EE mylists Q =I 


D WLY 

2A MECU Se 

21 neey 

4) "two" 

5) "one" 

redis 127.0.0.1:6379> lpop mylist3 
"fiven 

et Kee El 
de eene 

Zl three 

3) "two" 

4) "one" 

(7) rpop 


从 指定 的 list git (eat) 到 上 单个 弹出 《删除 ) 元 隶 。 如 果 执 行 成 功 将 
返回 被 弹出 的 元 系 名 称 ， 人 奋 则 返回 nil。 如 以 下 代码 所 示 。 
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SN 127-0.-0.156379> Llrange mylists Q =I 


EE 

2) three 

BK E 

4) "one" 

redis 127.0.0.1:6379> rpop mylist3 

Tone" 

recis E lranee mylists 0 =I 
Ee RE 

Zl ftbhreen 

3) "two" 


redis 127.0.0.1:6379> 
(8) rpoplpush 
从 指定 的 list 链接 列表 中 由 上 到 下 顺序 弹出 单个 元 素 ， 并 将 该 元 素 妃 加 到 新 链接 列表 的 
头 部 。 如 以 下 代码 所 示 。 


EE 127.0.0.136379> Lrange mylistá © =1 


ID A VASA 

EE 

21 DtCbhreeit 

4) "two" 

5) Yone“ 

recis 127:-0,:0.136379> lramee mylists 0 =I 
DIEESE 

Sch, A 

S SE 

AN 2 

E 

redis 127.0.0.1:6379> rpoplpush mylist4 mliseth 
"Tone" 

recis 127:-0:0.136379> lramee mylists 0 =I 
1) vone” 

Se SS 

SI 4 

4 DS 

SE 

EE 

(9) lindex 


根据 元 素 索引 号 ， 返 回 该 元素 内 容 。 如 以 下 代 但 所 未 。 


EE EE EE 


e Ee 

2I EE SCH 

5) eh 

4) "two" 

redis 127.0.0.1:6379> lindex mylist4 3 
"two" 

(10) llen 


返回 指定 list 链接 列表 的 长 上 度 〈 即 元 素数 量 )， 如 以 下 代码 所 示 。 
redis 127.0.0.1:6379> llen mylist4 

(integer) 4 

ECLS 127:0, 0, 11863179S 
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16.2.4 Sets 类 型 


在 数学 中 ， 集 合 表示 一 组 具有 某 种 共同 性 质 的 数学 元 素 。 在 现实 世界 中 ， 集 合 表 示 基 有 
某 种 特定 性 质 事物 的 总 体 。 在 Redis 中 ， 集 合 的 概念 与 数学 中 的 集合 是 一 样 的 。 使 用 集合 运 
算 ， 就 可 以 轻松 地 实现 好 友 推 荐 、 微 博 粉 丝 〈 关 注 )、 文 章 Tag 等 应 用 功能 。 对 集合 的 操 
作 ， 篆 见 的 有 并 集 、 交 集 及 差 集 。 事 实 上 ， 这 些 概念 也 是 传统 数据 中 的 集合 概 您 。Redis 使 
用 Hash table 实现 集合 间 的 差 集 、 并 集 、 区 集 等 钊 规 操作 。 接 下 来 将 结合 示例 代码 ， 话 细 介 
绍 集合 的 使 用 。 

1. 集合 的 常规 操作 

在 单个 集合 中 ， 第 见 操作 包括 添加 元 素 、 奏 看 元 素 、 删 除 元 素 、 弹 出 元 素 等 操作 。 这 些 
操作 都 有 对 应 的 命令 ， 下 面 分 别 介绍 。 

(1) sadd 

sadd 命令 用 于 问 指 定 的 集合 中 添加 元 素 。 如 果 指 定 的 集合 不 存在 ， 则 首先 创建 该 集合 ， 
然后 再 执行 诬 加 操作 。 需 要 注意 的 是 ， 在 集合 中 不 允许 添加 相同 内 容 的 集合 元 素 。 诡 加 成 功 
后 ， 返 回 结束 为 1， 否则 返回 0， 如 以 下 代码 所 示 。 


EE EE Sl "ese 
(integer) 1 

redis 127.0.0.1:6379> smembers s1 

E enpaw 

EONS 127:-0:-0-1;6379> sacc Sl celba 
(integer) 0 


(2) smembers 
HSPA, PEH smembers 碍 看 指定 集合 中 的 所 有 集合 元 素 。 运 行 成 
功 后 ，smembers 将 日 动 对 结果 进行 排序 。 如 果 指 定 的 集合 不 存在 ， 则 返回 空 值 。 如 以 下 


代码 所 示 。 

redis 127.0.0.1:6379> smembers s1 

d EE RE 

EE 

(3) arem 

根据 集合 名 称 ， 删 除 指定 集合 中 的 元 素 。 命 令 成 功 运 行 后 返回 1， 人 否则 返回 0。 如 以 下 
代码 所 示 。 

SOLS 127,.0.0. L639> srren sl ceiioas 

(integer) 1 

SG 下 127.-0-0.-1:6379> Srem sil ce 

(integer) 0 

recs 127:0:0.,1s 6379% 

(4) spop 


以 单条 的 形式 随机 弹出 删除 》 指 定 集合 中 的 元 素 。 如 果 指定 的 集合 不 存在 ， 将 返回 
mil， 否 则 返回 被 弹出 的 集合 元 素 。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> smembers s2 


L) Htbhreen 
lech 
EE 
4) "one" 
D VEOLIA 
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EE 8S2 
EE) 

EE le 
(nil) 


(5) smove 
将 指定 的 集合 元 素 移 动 到 新 的 集合 中 ， 该 命令 共有 3 个 参数 : 参数 1 为 源 集合 ， 参数 2 
为 目标 集合 (如 果 不 存 在 ， 则 目 动 创建 )， 参 数 3 为 被 移动 的 集合 元 素 。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> smembers s7 


I) Wom 
2) "two" 
3) VrO N 
A Trbhreei 
DI VELYyGY 


redis 127.0.0.1:6379> smove s7 s8 one 
(integer) 1 
redis 127.0.0.1:6379> smembers s8 


Ji We 

recs 1127-0-03 6379> smembers si 
H TEORY 

2) Venres” 

S) VCVO“ 

4) "five" 

recie 1127-0-0, 1L 63179> 

(6) scard 


退回 指定 集合 元 素数 量 。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> smembers s7 


D UEO 
2) "Trtbreei 
3) EGG 

Ze 


rechs 127.0.0,1306379> KH 

(integer) 4 

(7) sismember 

检测 集合 是 否 存在 指定 的 元 素 。 如 有 存在 则 返回 1， 否 则 返回 0。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> smembers s7 


IO EE OUN 
2) "Trtbhreei 
3) VTWOÙ 

4) "five" 


redis 127.0.0.1:6379> sismember s7 七 WO 
(integer) 1 

cecks 127-0:-0,136379> sistemos r s7 J 
(integer) 0 


(8) srandmember 


BEILA WIR ERS PIIR. WA PRATE. 


redis 127.0.0.1:6379> smembers s7 


IH VEON 
2) ieee 
9) Wiewe, 

4) "five" 


redis 127.0.0.1:6379> srandmember ol 
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ID four" 
redis 127.0.0.1:6379> srandmember s7 
"three" 


2. 差 集 、 并 集 、 交 集运 算 

两 个 集合 进行 对 比 后 ， 得 到 不 相同 的 数值 ， 就 称 为 差 集 。 同 样 ， 两 个 集合 对 比 后 ， 得 到 
相同 的 数值 就 称 为 交集 。 而 两 个 集合 对 比 后 ， 所 有 数值 相 加 束 称 为 并 集 。 假 设 有 两 个 集合 ， 
如 图 16-3 所 示 。 


图 16-3 集合 


如 有 果 要 对 图 16-3 所 示 集 合 进行 产 集 运算 ， 那 么 络 末 将 为 1、2 (Setsl 为 运算 集合 ， 
0 


Sets2 为 被 运算 集合 );， 交集 运算 的 结果 为 3; 并 集运 算 的 结果 为 1、2、3、4、5、6。 在 
Redis 集合 运算 中 ， 同 样 提 供 了 相应 的 命令 。 接 下 来 将 分 别 介绍 。 

(1) sdiff 

HATRI ERA AB 1 作为 运算 集合 ， 参 数 2 作为 被 运算 集合 )。 如 以 下 
代码 所 示 。 

redis 127.0.0.1:6379> smembers s3 

L) Vrnree™ 

2) EWON 

9) VON Y 

recheas 1127-0-01; 6379> smembers si 

TD TEON 

2) "Trtbreei 

3) fey 

redis 127.0.0.1:6379> sdiff s3 s4 

I) vong 

2) "two" 

运算 集合 与 被 运算 集合 是 可 以 互相 转换 的 。 得 到 的 结束 将 以 运算 集合 为 准 ， 如 以 下 代码 
Dro, 

redis 127.0.0.1:6379> smembers s3 

L) WEEw 

2) "two" 

3) VONG% 

redis 127.0.0.1:6379> smembers s4 

ID VE DE RS 

2) Vrhree™ 

S) ey 

redis 127.0.0.1:6379> sdiff s4 s3 

EE 

2 eas 


pechs 2 0 ET 65319 
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(2) sinter 
对 所 有 指定 的 集合 进行 交集 运算 。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> smembers s3 


1) Wem 

2) "two" 

21 Donen 

redis 127.0.0.1:6379> smembers s4 
TD en he 

2) WElMEGEY 

EH 

EE El EE EE 53 64 
Ji Htbhreen 


(3) sinterstore 


对 所 有 指定 的 集合 进行 交集 运算 ， 并 将 结 来 复制 到 新 的 集合 中 《如 果 新 集合 不 存在 ， 


自 完 创建 )。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> smembers s3 


L) VYtharee™ 

E 

EE omey 

redis 127.0.0.1:6379> smembers s4 
1 On 

2) MEE 

中 ES 


redis 127.0.0.1:6379> sinterstore s5 s3 s4 
(integer) 1 

cechas 1127-0-0, 136379> samembers 65 

1) Htbhreen 


上 述 代码 表示 将 集合 s3 与 s4 的 交集 运 握 结果 为 存 为 集合 s5 中 的 元 素 。 


(4) sunion 
对 所 有 指定 的 集合 进行 并 集运 算 。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> Sunion s3 s4 


teure 
2) "two" 
SE 
di Eee 
Sy kN 


5) snionstore 


对 所 有 指定 的 集合 进行 并 集运 算 ， 并 将 结 来 复制 到 新 的 集合 中 《如 果 新 集合 不 存在 ， 


自 完 创建 )。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> smembers s3 


L) VEarSeY 

2) "two" 

5) Omey 

redis 127.0.0.1:6379> smembers s4 
IA LEOUN S 

2) Trtbhreei 

Ee Ee 


redis 127.0.0.1:6379> sunionstore s7 s3 s4 
(integer) 5 
redis 127.0.0.1:6379> smembers ai 


1) "one" 


则 


则 
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2) "two" 

S390 Ow 
di Trbhreei 
E EEN 


EE 0. 0. 1: 631797 


16.2.5 Zset 类 型 


Sets 集合 类 型 是 一 种 没有 排列 顺序 的 集合 ， 开 发 人 员 只 能 通过 添加 的 前 后 顺序 进行 拓 
序 。 并 且 在 查询 数据 时 ， 只 能 使 用 元 素 内 容 进行 查询 ， 而 不 能 通过 索引 进行 查询 。 如 果 集合 
内 容 是 一 串 长 文本 ， 显 然 这 种 查询 方式 是 比较 烦 项 的 。Zset 是 一 种 增强 型 的 Sets， 也 称 有 序 
集合 。 顾 名 思 义 就 是 在 集合 中 添加 排序 。 如 图 16-4 所 示 。 


图 16-4 有 排序 的 集合 


此 外 ，Zset 在 添加 集合 元 系 时 ， 系 统 会 目 动 为 每 个 元 素 分 配 上 唯 一 的 家 引 号。 开发 人 员 可 
以 通过 唯一 索引 号 取得 对 应 的 集合 元 素 内 容 。 在 操作 上 与 List 列表 相 类 似 。 接 下 来 将 对 Zset 
第 见 操作 进行 评 细 介绍 。 

(1) zadd 

Il Zset 中 添加 元 素 。 如 果 添 加 成 功 则 返回 1， 人 否则 返回 0。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> zadd z1 1 one 


(integer) 1 
ecis 127:-0-0,.15:6379> zaco zi 2 TWO 
(integer) 2 
EON 127-0,:0,.136379> zaco zi 1 One 
(integer) 0 
recis 127.0 
(2) zrange 
在 指定 的 范围 内 查询 指定 的 Zset 数据 集合 ， 并 按照 排序 号 自动 完成 排序 〈 由 小 到 大 )。 
该 命令 文 持 2 个 参数 ， 分 别 表示 列表 查询 冰 围 (从 上 到 下 由 0 开始 ，-1 表示 列表 最 后 一 个 元 
素 )。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> zrange z1 0 -1 


-0o L3 > 


1L) "one" 

2) ee 

3) "two" 

如 果 需 要 显示 排列 顺序 号 ， 只 需要 在 命令 后 加 上 withscores 标记 即 可 ， 如 以 下 代码 
Dro, 

redis 127.0.0.1:6379> zrange zl 0 -1 withscores 

1) "one" 

2) wW w 
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3 ) "two" 
4) UD W 

D MENTSEN 
6) W RW 


(3) zrevrange 

在 指定 的 范围 内 得 询 指定 的 Zset WRES, FARAT S ASIRNA KRENA). 
该 命令 支持 2 个 参数 ， 分 别 表示 列表 查询 范围 《从 上 到 下 由 0 开始 ，-1 表示 列表 最 后 一 个 元 
K)o UWM PIRIT. 


ecis 127-00-01 6379> zrevrangs z5 Q = wLrnscores 


T ETE ASA 
E ESR 

DI MEOLA 
4) "An 

S) ee 
INT WDN 

7) "two" 
B) SZU 

OI omey 
NET 


(4) zrevrank 


返回 指定 Zset (DIr Wide, HRAKIN B SHEF. WA PRITE 


EONS 127-0:-0,136379> zrange z4 0 =L wicascores 


1) "two" 

2) WD W 

3) "one" 

4) Wew 

5) “tehreeT 

6) W E WU 

Toe 

8) A ai 

9) "five" 
10) WW 


redis 127.0.0.1:6379> zrevrank z4 three 
(integer) 2 


(5) zrangebyscore 


根据 Zset SC OI, BERS PIKIR. WA PRIIT o 


els 127:-0.0,.136379> zrange 25 Q =I 


1) vone” 

2) "two" 

5) Miele 

4) "four" 

Sy VTLS 

redis 127.0.0.1:6379> zrangebyscore z5 3 4 
L) TrCbhreei 

EE 

(6) zcount 


根据 Zset 索引 写 ， 统 计 集 合 中 的 区 间 元 素数 量 。 如 以 下 代码 所 示 。 


CLS 1276060 EE z5 Q =I 


1) "one" 
2) "two" 
3) "three" 
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on, 

CH 

EE EE 
(integer) 3 


(7) zcard 
统计 指定 Zset TRA. WLA FRITZ. 


cecis EE zrange z5 0 =1 


Ji Vong 
2) won 
21 nnaeey 
AD WE EE eeh 
S AR ASN 


recie 1127-0-0. 1:6379> zeA ZS 
(integer) 5 


(8) zrem 
根据 元 素 名 称 ， 删 除 指定 Zset 中 的 集合 元 素 。 如 果 删 除 成 功 则 返回 1, EURE 0。 如 
以 下 代码 所 示 。 


Elle wam Zl OnE 
(integer) 1 

GCE 127:0:0:126379> zrange zL © =i 
L) TrCbhreei 

2) "two" 


(9) zincrby 
根据 指定 的 元 素 名 称 ， 对 该 元 妹 排 序 进行 加 值 运 算 。 如 以 下 代码 所 示 。 


Lecke 1127-0-01; EE H z4 e REES 


1) WAMEY 
PN WIRU 

3) "two" 
AN 

5) "three" 
G) NST 

T) WAMWEY 
SANAN 
Girres 
SIb ASA 
redis 127.0.0.1:6379> zincrby z4 2 one 
dE: 

redis 127.0.0.1:6379> zrange z4 0 -1 withscores 
1) "two" 
DVA ZN 

3) "one" 
4) "3" 

5) "three" 
O TSN 

WD HEOI W 
8) "4" 
EE 
EE 


(10) zremrangebyscore 
根据 排序 号 ， 对 区 间 内 的 数据 进行 批量 删除 。 执 行 成 功 后 ， 将 返回 被 删除 的 数据 量 。 如 
以 下 代码 所 示 。 
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EE L27-0:0,156379> zrange z7 EE 


L Vong“ 
E IFOR 
S33 Eweov 
E 
5) "three" 
S VIZA 
TO EO 
S TESA 
9 EE 
E Bee 


redis 127.0.0.1:6379> zremrangebyscore z7 10 13 
(integer) 4 

techies 127:-0-0,.156379> zrange zl Q =L wiLtrascoreg 
I EE 

2) UTAN 


如 上 述 标 粗 代 人 码 所 示 ， 表 示 删 除 z7 集合 中 排列 顺序 10 一 13 区 间 内 的 数据 。 如 果 指 定 的 
排序 号 不 存在 ， 则 返回 0。 

(11) zremrangebyrank 

根据 唯一 索引 写 ， 对 区 间 内 的 数据 进行 批量 删除 。 执 行 成 功 后 ， 将 返回 被 删除 的 数据 
量 。 如 以 下 代码 所 示 。 


recis 127.050.18063793> zrange z9 Q =1 


Ji Tonei 
EE 
21 ES 他， 
ANE DER e 
H 


redis 127.0.0.1:6379> zremrangebyrank z9 1 3 
(integer) 3 
escis 127,00,123 6379> zrange 29 0 =1 


E Eet 

EE El 

如 末 对 单个 元 素 进行 删除 ， 上 只 需要 为 两 个 参数 传 入 相同 的 索引 号 即 可 。 如 以 下 代码 
Dro, 


redis 127.0.0.1:6379> zremrangebyrank z9 1 1 
(integer) 1 


16.2.6 ”使 用 phpRedisAdmin 


redis-cli 管理 终 闹 是 Redis 官方 提供 的 基于 命令 行 的 管理 工具 ， 能 够 全 面 提供 对 Redis 的 
文 持 。 如 采 谈 者 不 习惯 使 用 命令 工具 ， 当 然 也 可 以 选择 可 视 化 的 管理 工具 ， 例 如 
phpRedisAdmin 就 是 一 敌 使 用 简单 、 开 源 及 免费 的 Redis 在 线 管 理 系统 ， 在 使 用 方式 上 与 
phpMyAdmin 其 似 。 接 下 来 首先 介绍 phpRedisAdmin 的 安装 。 

1. phpRedisAdmin 的 安装 

phpRedisAdmin 是 基于 PHP 编写 的 一 套 Web 软件 ， 所 以 并 没有 安装 过 程 。 虽 然 如 此 ， 
ZRNA ELE phpMyAdmin Ze (Ch oi. A nl DL dr https://github.com/ErikDubbelboer/php 
RedisAdmin 项 目地 址 中 找到 对 应 的 源 代码 压缩 包 ， 这 里 下 载 的 版 本 为 2.4.17。 读 者 也 可 以 在 本 
书 配套 的 文 持 网 站 中 下 载 ， 地 址 为 http://beauty-soft.net/book/php_mvc/down/phpredisadmin.html。 
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打开 includes/config.inc.php 配置 文件 ， 根 据 实际 环境 修改 其 中 的 配置 信息 。 如 以 下 代码 
ra, 


<?php 
Şconfig = array ( 
'servers' => array ( 


0 => array ( 

'name' => 'local server', // 显示 名 称 

'host' => '192.168.0.2', //redis 服务 器 地 址 

'port' => 6379, //redis 通信 端口 

'filter' => '*'，// 显 示 的 字段 

'auth'=>'123456'  ”/// 登 录 密 码 

// Optional Redis authentication. 

//"'auth' => 'redispasswordhere' // Warning: The password is sent in plain-text 
Co Che RECS server, 


) y 


/*1 => array ( 
host” => "localhost", 
Woort => 6380 

E 


/*2 => array ( 
aame! => “local dio 2, 
host” => "localhost", 
Woort" => 63179; 
vela” => 1 // Optional database number, see http://redis.io/commands/select 
'filter' => 'something:*' // Show only parts of database for speed or security 


reasons 
E 
) y 
SCOPEerLauor ee > 
// Uncomment to show less information and make phpRedisAdmin fire less commands to 
the Redis server. Recommended for a really busy Redis server. 
TV e E E 
// Uncomment to enable HTTP authentication 
/= t login" => array l 
7 Username > Password 
// Multiple combinations can be used 
'admin' => array ( 
'password' => 'adminpassword', 


) v 


'guest' => array ( 
'password' => II, 
'servers' => array(1) // Optional list of servers this user can access. 
) 
ee 


// You can ignore settings below this point. 


'maxkeylen' => 100 


i 
?> 


保存 config.inc.php 文件 ， 直 接 访 问 phpRedisAdmin 所 在 网 址 即 可 。 界 面 如 几 16-5 
所 示 。 
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| Firefox ~ i 
@ 192.168.0.2 - phpRedisAdmin |+ 


图 16-5 phpRedisAdmin 界面 
如 果 读 者 并 非 在 本 书 配套 网 站 中 下 载 phpRedisAdmin， 直 接 访问 将 会 提示 找 不 到 Redis 
连接 类 库 。 如 图 16-6 所 示 。 


Firefaox = 


+e >» UG e localhos %7 7 e. Goc P| A D- ex 


Warning: require_once(DADownloads\xampp-win32-1.7.4-VC6\xampp 
\htdocs\phpRedisAdmin/predis/autoload.php) [function.require-once]: 
failed to open stream: No such file or directory in DADownloads 

\wampp-win32-1.7.4-VC6\xamppihtdocsiphpRedisAdmin\includes 


\common.inc.php on line 31 


Fatal error: reoure oncel0 [function.require]: Failed opening required 
‘DADownloads\xwampp-win32-1.7.4-VC6Wwxampp\htdocs 
“phpRedisAdmin/predis/autoload.php’ (include path=".:DA\Downloads 
‘wampp-win32-1.7.4-VCBwamppphA\PEARY in DADownloadsixampp- 
win32-1.7.4-VCõ6\xamppihtdocs\phpRedisAdmin\includes 
\wcommon.inc.php on line 31 


图 16-6 k/b Redis 类 库 


解决 办 法 : 只 需要 使 用 git 安装 Redis 类 库 即 可 (Windows 环境 下 可 以 使 用 可 视 化 的 
GitHub). git 地 址 为 https://github.com/ErikDubbelboer/phpRedisAdmin.git。 以 下 是 Linux 环境 
下 的 安装 过 程 。 


[root@lbogon ~]# cd /home/wwwroot/phpRedisAdmin 


[root@ phpRedisAdmin]# yum install git* 

[root@ phpRedisAdmin]# git clone https://github.com/ErikDubbelboer/phpRedisAdmin.git 
[root@ phpRedisAdmin]# cd phpRedisAdmin 

[root@ phpRedisAdmin]# git submodule init 

[root@ phpRedisAdmin]# git submodule update 


2. phpRedisAdmin 的 使 用 
在 phpRedisAdmin 中 进行 数据 管理 ， 将 变 得 非常 直观 。 例 如 创建 一 个 Key， 只 需要 单 击 
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左 侧 导 航 栏 中 的 “Add another key” 链 接 。 在 弹出 的 创建 Key 页 面 中 ， 可 以 选择 Key 类 型 、 
名 称 、 内 容 等 。 如 图 16-7 所 示 。 


Add 

Types (String 
Key: 

wahia: | 


图 16-7 创建 Key 表单 


16.3 Redis 高 级 使 用 


Redis 文 持 丰 曲 的 功能 特性 ， 充 分 使 用 这 些 特性 将 让 Web 应 用 更 加 稳定 及 高 效 。 接 下 来 
将 分 别 介绍 数据 持久 化 、 虚 拟 内 存 、 事 务 处 理 等 局 级 的 实用 功能 。 


16.3.1 数据 持久 化 


前 面 已 经 多 次 提 到 过 ，Redis 是 一 个 内 存 数据 库 。 但 Redis 与 Memcache 特别 之 处 在 于 提 
供 了 数据 持久 化 功能 ， 使 得 数据 更 加 安全 及 完整 ， 也 为 大 容量 储存 提供 了 可 靠 文 持 。Redis 
共 文 持 两 种 数据 持久 化 方式 ， 分 别 为 SnapShotting 方式 与 Append-only file 方式 。 下 面 分 别 介 


给 。 


1. SnapShotting 

SnapShotting 方式 即 快照 方式 。 这 是 Redis 默认 使 用 的 数据 持久 化 方式 。 这 种 方式 是 将 
内 存 中 的 数据 以 快照 的 方式 写 入 dum.red 二 进 制 文件 中 ， 我 们 可 以 通过 配置 redis.conf 文件 ， 
改变 快照 的 执行 规则 及 时 间 。 如 以 下 代码 所 示 。 

save 900 1 


save 300 10 
save 60 10000 


如 上 述 代码 所 示 ， 其 中 第 1 条 规则 表示 900s 内 如 果 超过 1 条 Key 记录 被 修改 ， 则 执行 
快照 记录 。 依 次 类 推 ， 第 2 条 和 第 3 条 规则 的 作用 也 是 一 样 的 。 
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2. Append-only file 

Append-only file MJÆR Aof。SnapShotting 方式 虽然 在 一 定 程度 上 解决 了 数据 持久 化 的 问 
题 ， 但 由 于 是 基于 时 间 规 则 来 实现 的 ， 所 以 还 会 存在 数据 丢失 的 可 能 。 例 如 系统 在 快照 规则 
内 出 现 故 障 ， 这 时 在 内 存 中 的 数据 将 被 清除 。Aof 的 智能 之 处 就 在 于 所 有 操作 都 是 基于 
Write 函数 来 实现 的 。 也 束 是 说 Redis 在 执行 号 操作 时 ， 都 是 会 收 到 操作 系统 内 核发 出 的 指 
令 ， 然 后 通过 Write AAE Jun 2 appendonlyaof 文件 中 。Aof 默认 是 关闭 的 ， 即 


appendonly no。 如 果 和 需要 打开 Aof， 上 只 需要 将 appendonly 设置 为 yes 即 可 ， 如 以 下 代码 所 未 。 

appendonly yes # 开 局 Aof 

appendfsvnc alwaus Aoa UEN 

#appendfsync everysec # 每 秒 执行 一 次 

#appendfsync no # 使 用 操作 系统 内 核 指令 

如 上 述 代 码 所 示 ，Aof 同步 方式 共 分 为 3 种 方式 ， 分 别 如 下 。 

> appendfsync always: 基于 应 用 层面 触发 Write 函数 ， 实 现 立 即 执行 数据 同步 操作 ， 但 

速度 轻 慢 。 

> appendfsync everysec: 通过 时 间 触 发 函数 实现 数据 同步 ， 速 度 一 般 。 

> appendfsync no: 基于 操作 系统 内 核 监控 指令 ， 速 度 较 好 。 

需要 注意 的 是 ，appendfsync no 方式 虽然 在 性 能 上 比较 好 ， 但 由 于 某 些 操作 系统 会 对 
Write 图 数 操作 进行 缓存 ， 所 以 并 不 会 立即 执行 同步 操作 ， 此 时 也 会 存在 数据 丢失 的 风险 。 


16.3.2 ”虚拟 内 存 


内 存 数 据 库 最 大 的 优点 融 是 数据 谈 写 速度 快 ， 但 缺点 也 非常 名 显 。 其 中 容量 驶 是 一 个 随 
时 需要 面 对 的 问题 。 尽 管 近年 来 ， 内 存 容量 已 经 有 了 大 幅 提 升 ， 价 格 也 已 经 变 得 非常 便宜 。 
但 与 便 盘 比 起 来 ， 无 论 是 在 容量 还 是 价格 方面 ， 便 盘 都 具有 不 可 和 奉 代 的 地 位 。 所 以 将 一 部 分 
数据 ， 例 如 不 常用 的 数据 或 者 较 老 旧 的 数据 转 存 到 便 盘 上 ， 是 一 种 廉价 及 局 效 的 储存 方案 。 

在 Redis 中 虚拟 内 存 就 是 这 样 的 一 种 技术 ， 它 的 主要 作用 就 是 将 不 经 常 使 用 的 数据 转 存 
到 硬盘 上 ， 这 样 既 确保 了 读 写 速度 ， 又 很 好 地 解决 了 容量 问题 。 要 开启 虚拟 内 存 功能 ， 只 需 
要 在 配置 文件 中 将 vm-enabled 选项 设置 为 yes 即 可 《默认 为 no)， 如 以 下 代码 所 示 。 

vm-enabled yes # 开 局 虚拟 内 存 

vm-swap-file /tmp/redis.swap # 虐 拟 内 存 交 换文 件 

vm-max-memory 1000 #redis 使 用 内 存 上 限 ， 字 节 为 单位 

vm-page-size 32 # 每 页 大 小 ， 字 而 为 单位 

vm-pages 134217728 # 最 多 交换 页 面 数量 

vm-max-threads 4 # 执 行 交 换 任务 的 线程 数量 


其 中 vm-max-threads 表示 执行 虚拟 内 存 交 换 任 务 的 线程 数量 ， 在 实际 应 用 中 一 般 需 要 根 
据 CPU 数量 进行 设置 (CPU 数量 *4)。 


16.3.3 事务 处 理 

Redis 是 一 个 为 数 不 多 支持 事务 处 理 的 NoSQL 数据 库 系统 。 事 务 处 理 不 是 必需 的 ， 但 它 
对 数据 的 可 控 性 非常 有 效 ， 这 在 一 些 对 安全 性 、 完 整 性 较 高 的 场合 中 是 非常 必要 的 。redis-cli 
使 用 multi 命令 实现 事务 处 理 。 

事务 处 理 的 一 般 流程 用户 提 交 的 操作 语句 会 被 保存 到 队列 中 ， 直 到 发 出 显 式 的 执行 命 
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令 ， 系 统 才 会 真正 执行 操作 。 在 等 待 显 式 的 执行 命令 之 前 ， 用 户 可 以 进行 任何 的 数据 操作 ， 
例如 增 、 删 、 改 、 碍 。 事 务 机 制 会 对 操作 语句 进行 容错 性 检测 ， 如 果 存 在 其 中 一 条 异常 操 作 
语句 ， 系 统 将 放弃 队列 中 的 操作 ， 称 为 事务 回 深 (也 可 显 式 声明 事务 回 深 )。 

在 MySQL 数据 库 中 ， 使 用 begin 命令 声明 一 个 事务 ; 使 用 commit 声明 事务 提交 ; 使 用 
rollback 声明 事务 回 深 。 而 在 Redis 中 ， 同 样 提 供 了 相应 的 事务 处 理 命 令 ， 分 别 为 multi GS 
明 事 务 )、exec〔 提 交 事 务 )、discard (事务 回 深 )。 接 下 来 将 结合 示例 ， 演 示 Redis 事务 处 理 
功能 。 

redis 127.0.0.1:6379> multi 

OK 

redis 127.0.0.1:6379> hset bookl author ceiba 

QUEUED 

redis 127.0.0.1:6379> hset book2 author beauty-soft 

QUEUED 


redis 127.0.0.1:6379> exec 
1) (integer) i. 


2) (integeri 1 

recheas 127:0.0, 18 6379> Me Oooki Eeer 
Yosia” 

redis 127.0.0.1:6379> hget book2 author 
"beauty-soft" 


上 述 代 人 码 分 别 使 用 multi 事务 实现 了 两 个 Hash Key 的 创建 。Redis 使 用 discard 处 理事 务 
ERR. aA FREI 


redis 127.0.0.1:6379> hget bookl1 author 
"ceiba" 

redis 127.0.0.1:6379> hget book2 author 
"beauty-soft" 

recie 1127-0-0. 156379> mule 

OK 

redis 127.0.0.1:6379> hset book1l author ceibas 
QUEUED 

redis 127.0.0.1:6379> hset book2 author mysoft 
QUEUED 

redis 127.0.0.1:6379> discard 

OK 

redis 127.0.0.1:6379> hget bookl1 author 
"ceiba" 

redis 127.0.0.1:6379> hget book2 author 
"beauty-soft" 


可 以 看 到 ， 虽 然 在 事务 中 ， 提 交 了 两 条 hset 操作 指令 ， 但 由 于 使 用 了 discard 命令 ， 所 
以 最 终 的 结 末 将 取消 所 有 事务 队列 中 的 操作 指令 。 

需要 注意 的 是 ，Redis 对 事务 的 文 持 非 钊 有 限 。 例 如 在 传统 的 关系 型 数据 库 中 ， 如 末 队 
列 中 存在 一 条 错误 的 SQL， 事 务 机 制 将 放弃 队列 中 的 所 有 操作 。 而 在 Redis H, FAR H] 
样 严格 的 事务 机 制 。 如 以 下 代码 所 示 。 


redis 127.0.0.1:6379> hmset bookl1l author ceiba price 62 
OK 

EONS 127:-0:0.-1:6379> Mee Dook 于 二 各 

"gom 

gecis 127:-0:0,1363579> milti 

OK 
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redis 127.0.0.1:6379> hincrby booklL price 3 

QUEUED 

redis 127.0.0.1:6379> hincrby bookl price sixty 

QUEUED 

cecs EE 

1) (integer) 65 

2) (error) ERR value is not an integer or out of range 
"EON 127:-0:0,136379> hger booki Price 

nenn 


如 上 述 标 粗 代码 所 示 ,“hincrby bookl price 3” 是 一 条 正确 的 指令 ， 放 到 事务 中 是 能 够 
执行 的 ; 但 “hincrby book) price sixty” 明 显 是 一 条 馈 误 的 指令 (hincrby A SC rr ER A, Pr 
以 在 执行 exec 命令 时 将 提示 “(error) ERR value is not an integer or out of range”. 根据 传统 的 
关系 型 数据 库 事务 机 制 特 性 ， 只 要 队列 中 存在 异常 的 执行 语句 ， 将 放弃 队列 中 所 有 指令 “〈 即 
捉 务 回 深 )。 所 以 尽管 “hincrby bookl price 3” 指 令 正 确 ， 但 最 终 的 结果 理应 是 不 产生 作用 
的 。 但 实际 情况 是 ，Redis 执行 了 操作 。 从 这 个 角度 上 看 ，Redis 中 的 事务 功能 是 不 完善 的 ， 
至 少 与 我 们 常见 的 事务 机 制 存在 观念 上 的 区 别 。 这 点 在 实际 应 用 开发 中 ， 需 要 读者 目 行 区 分 
(不 排除 后 续 版 本 会 进行 完善 )。 


16.3.4 主 从 同步 


主 从 同步 是 进行 读 写 分 离 的 首要 前 提 〈 读 写 分 离 的 意义 详 见 本 书 7.5.2 市 )，Redis LA 
多 种 数据 主 从 同步 方式 ， 包 括 一 主 多 从 、 一 主 一 从 等 。 在 配置 方式 上 与 传统 的 MySQL 相 比 
显得 非常 简单 。 只 需要 在 配置 文件 中 开启 slaveof 选项 即 可 。 接 下 来 将 通过 示例 ， 详 细 介绍 
一 主 一 从 的 同步 方式 。 

这 里 将 使 用 192.168.0.2 服务 器 作为 主 Redis 数据 库 ;， 192.168.0.3 作为 从 Redis 服务 
器 。 首 先 将 主 服务 器 中 的 Redis 复制 到 从 服务 占 中 ， 以 便 两 台 服 务 占 的 Redis 数据 相同 。 
mes 


[root@bogon bin]#scp -r /usr/local/redis/ root@192.168.0.3:/usr/local/redis 


root@192.168.0.3's password: 

输入 正确 的 系统 密码 ， 即 可 进行 复制 。 如 果 sp 命令 不 可 用 ， 只 需要 安装 openssh- 
clients 即 可 ， 命 令 如 下 。 

[root@~]#yum -y install openssh-clients 

复制 完成 后 ， 可 以 在 192.168.0.3 服务 器 中 查看 Redis 源 代码 文件 ， 这 些 文 件 与 
192.168.0.2 服务 右上 的 文件 是 一 样 的 。 

完成 上 述 操作 后 ， 修 改 192.168.0.3 服务 器 上 的 Redis 配置 文件 即 可 ， 如 以 下 代码 所 示 。 


slaveof 192.168.0.2 6379 


If the master is password protected (using the "requirepass" configuration 
directive below) it is possible to tell the slave to authenticate before 
starting the replication synchronization process, otherwise the master will 
refuse the slave request. 


S Gs ua un Si 


masterauth 123456 


标 粗 的 即 为 需要 修改 的 配置 项 。 完 成 后 保存 redis.conf 文件 ， 局 动 Redis 服务 即 可 。 
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[root@~]#ce /usr/local/redis/bin 

[root@bogon bin]# ./redis-server /usr/local/redis/etc/redis.conf 

通过 前 面 的 操作 ， 一 主 一 从 的 一 个 Redis Eft aCHk T, old 192.168.0.2 服务 器 上 进 
ÍT Redis 数据 操作 ， 例 如 增加 Key 或 删除 Key; 然后 观察 192.168.0.3 服务 器 上 的 Redis 数据 
变化 情况 。 在 192.168.0.2 主 Redis 服务 器 中 使 用 info 命令 ， 可 以 但 看 当前 Redis 状态 。 


16.4 Æ MVC 中 使 用 Redis 


国外 许多 主流 的 PHP MVC 框架 都 提供 了 Redis 编程 功能 ， 例 如 CakePHP, Akelos 等 。 
接 下 来 继续 以 国内 较 车 名 的 ThinkPHP 为 例 ， 详 细 介 绍 Redis 的 CURD 操作 。 在 此 之 前 ， 首 
先 需要 确保 当前 环境 已 经 正确 安装 了 Redis for php 扩展 (可 参阅 本 书 10.3.2 节 )。 为 了 方便 
介绍 ， 接 下 来 首先 介绍 在 传统 PHP 中 的 Redis 查询 操作 ， 帮 助 读 者 迅速 掌握 Redis for PHP 
API 的 使 用 。 


16.4.1 在 PHP 中 使 用 Redis 


Redis for PHP 扩展 分 为 儿 种 ， 例 如 Predis、phpredis、Rediska 等 。 每 种 扩展 在 使 用 方式 
上 不 尽 相 同 ， 读 者 可 以 在 http://redis.io/clients 网 页 上 找到 相关 的 API 说 明 。 一 个 最 简单 的 
phpredis API 的 示例 如 以 下 代 人 码 所 示 。 


<?php 


$redis = new Redis (); 
$redis->connect ('192.168.0.2', 6379); 
Ş$redis->auth="123456"; 
Ş$redis->set ('test','hello world!'); 
echo $redis->get ('test'); 

?> 


如 上 述 标 粗 代码 所 示 ， 表 示 问 test FFE Key 添加 “hello world!” 字 符 串 。 通 过 前 面 的 
学 习 ， 相 信人 读者 对 该 成 员 方 法 并 不 阳 生 。phpredis API 文 持 大 多 数 redis-cli 操作 命令 ， 并 日 保 
持 名 称 上 的 高 度 相 似 ， 方 便 PH 开发 人 员 进 行 操 作 ， 接 下 来 将 对 第 用 的 成 员 方法 进行 介绍 
《更 多 许 细 的 使 用 方法 可 参阅 https:/github.com/nicolasff/phpredis ) 。 

1. String 类 型 

(1) getSet 

返回 原来 Key KE, J value 与 入 Key。 


e EE E (Yx! A21) g 


SexValune = Srecdnis =>0Serc (Yx, “lolt) y 


snewValue = Ş$redis->get('x')"' 
(2) append 
在 指定 的 Key Value 后 面 妃 加 新 的 Value。 


Se DE EE "valuel") 7 


SE Se RE El EE 
Sredia-zoet ('key'); 


(3) getRange 
根据 指定 Key， 对 Value 进行 字符 截取 。 
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EE >sSec( key, EE 
SE EE (key, 0, E: 
Sredqis=>getRange ('key', -5, -1); 
(4) setRange 

b e b er > Di >> Zecbe xv AC 
根据 指定 Key, 对 指定 位 置 的 字符 进行 更 改 。 
SS EE “Hello worlel”) p 
Srel g=->setrRange (Y key"; 6, "recns")s 


$redis->get ('key'); 

(5) strlen 

获取 指定 Key Value JTI RKE. 
$redis->strlen ('key'); 

2. List 类 型 

(1) lPush 

EAZI AAIE ARIA o 


Şredis->lPush (key, value); 


(2) rPush 
EZI ERII. 


Şredis->rPush (key, value); 


(3) lPushx 
在 队列 的 头 部 顺序 添加 元 素 ， 如 果 指 定 的 Key 已 经 存在 ， 则 放弃 操作 。 


Şredis->lPushx (key, value); 


(4) lPop 

从 队列 的 头 部 依次 单个 弹出 元 素 。 
EE Ee ee 

(5) lSize 

统计 指定 List 元 素数 量 。 

SE le E Ee 


(6) lIndex 
根据 队列 索引 ， 返 回 相 应 的 元 素 内 容 。 


SS DE te EE 

(7) lSet 

根据 队列 索引 ， 设 置 对 应 的 元 系 内 容 。 

SE DEE (key O, Ee 

(8) (Range 

根据 区 间 索 引号 ， 碍 询 指定 Key 内 的 批量 元 素 《〈-1 表示 到 达 尾 部 最 后 一 个 元 素 )。 
Srechlg-> Range (keyi, 0, =1); 

(9) ITrim 

批量 删除 指定 List 数据 ， 但 你 留 start 及 end 参数 区 间 内 的 数据 。 


Sacls >l11Ireim( key", starce, enel) y 


(10) IRem 
从 指定 的 List 连接 表 中 批量 删除 相同 的 元 素 。number 为 0 时 ， 将 删除 所 有 元 素 。 


sredis->lRem('key', 'A', number); 
3. Hash 类 型 
(1) hSet 
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器 指定 的 Hash Ze De BE D Value。 如 来 指定 的 字段 已 经 存在 ， 则 执行 修改 操作 。 
Sredis— >nSerE (CML; lielclY, Yrest" iy 
(2) hGet 

查询 指定 Hash Key 中 指定 的 字段 值 。 
Srecdie-zbGer ('h1', 'field1l'); 

(3) hLen 

获取 指定 Hash Key 的 元 素数 量 。 
Ş$redis->hLen('h1'); 

(4) hDel 

删除 指定 Hash 字段 及 字段 值 。 

SE DEE ENEE eg e 

(5) hKeys 

列 出 指定 Hash Key 所 有 字段 。 
Şredis->hKeys('h1'); 

(6) hVals 

列 出 指定 Hash Key 所 有 字段 值 。 
$redis->hVals('h1'!) 

(7) hGetAll 

列 出 指定 Hash Key 所 有 字段 及 字段 值 。 
Ş$redis->hGetAl1l('h1'); 

(8) hExists 

检查 Hash Key E MTEI ERTE. 
$redis->hExists('hl', 'field1l'); 

(9) hIncrBy 

对 指定 的 Hash FRA 《数值 类 型 ) 进行 加 值 运算 。 


EE 


(10) bMset 

批量 添加 Hash 元 素 。 

Secns— >nveec ("neer il, array ("nane. => Teeba’, “salary => 2000) ) 7 
(11) hMGet 


根据 Hash FR, WEA AAI FEE 

Secls >nmeeE (C hnl, arcay(“rtieldil”, YrLieLldz2”)) s 
4. Sets 类 型 

(1) sAdd 


问 指 定 的 集合 顺序 添加 元 素 。 
Şredis->sAdd (key , value); 

(2) sRem 

Wir OAR), Mkieucnz, 
See ek 

See e 
SE EE 
SE 


(3) sMove 
在 指定 的 两 个 集合 中 ， 移 动 指定 的 集合 元 素 。 
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Şredis->sMove (seckey, dstkey, value); 


(4) sIsMember 
在 指定 的 集合 中 ， 检 得 是 否 存在 指定 的 元 素 。 


Şredis->sIsMember (key, value); 


(5) sCard 

获取 指定 集合 的 元 素数 量 。 

Secls—>sCare(" key") 

(6) sPop 

对 指定 的 集合 进行 随机 删除 元 素 。 

Şredis->sPop ('key') 

(7) sinter 

对 多 个 集合 进行 交集 运算 。 

Seclis >8Iner(array "sl", "82")) 

(8) sInterStore 

对 多 个 集合 进行 交集 运算 ， 并 将 交集 保存 到 output 的 集合 。 
Secls EE ee 


(9) SUnion 

对 多 个 集合 进行 并 集运 算 。 

Se D Bebe EE 

(10) sDiff 

对 多 个 集合 进行 差 集运 算 。 

SE "80", 1811, EE 

(11) sDiffStore 

对 多 个 集合 进行 差 集运 算 ， 并 将 差 集 保存 到 output 的 集合 。 
SE DEE EE GE Ehe EE 
(12) sMembers 

查询 指定 全 集 所 有 元 素 。 

Ş$redis->sMembers ('s1'!'); 

5. ZSet 类 型 

(1) zAdd 

在 指定 的 Zset 中 添加 排序 号 及 元 素 名 称 。 


EE 
Se ee ER KE e 
sredis -zAda E ee EE 
SS DEE Ee 


(2) zRange 

AW Zset PKI. RARER AIHEUT r ETHE 
SrecliLas->zAcel (Yzal; QO, VYOV)s 

ŞSrecdis=->zAdcc (“zl EE 

Şrecdis=->zAcda (tal, 10, “yi0” )g 

ŞrechiLg=>zRange (zil, 0, =1)7 


(3) zRem 


删除 指定 的 Zset 元 素 。 
SE EE 
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ŞSrecdis=>zRem( "zl", Vv!)p 

ŞrechiLe=>zRange (“zi 0, =1)7 

(4) zRevRange 

查询 Zset 中 的 区 间 元 素 ， 并 让 结果 是 否 显示 相应 的 排序 号 。 
SE Ee KE QO VYOV)s 

See E KEE 
GEI 

EE EE 0; =l; Crue)? 


(5) zCount 

获取 指定 Zset 元 系数 量 〈 文 持 区 间 获 取 )。 
Se ZEON 2 0 
zRemRangeByScore 


根据 排序 号 ， 删 除 指定 的 Zset 元 素 〈 文 持 区 间 删 除 )。 
Şredis->zRemRangeByScore('z1', 1, 5); 

(6) zCard 

查询 指定 Zset 元 素数 量 。 

sredis—>zCard('z1'); 


(7) zScore 
返回 指定 Zset 元 素 排 序号 。 


Ş$redis->zScore('zl', 'v1'!'); 


(8) zIncrBy 
对 指定 的 Zset UR ZUEK) HITIRI - 


$redis->zIncrBy('z2', 'age', 3); 


16.4.2 在 MVC 中 进行 CURD 操作 


接 下 来 以 ThinkPHP 作为 MVC 开发 框架 ， 详 细 介 绍 Redis 的 CURD 操作 。 需 要 说 明 的 
Æ» Œ ThinkPHP 中 本 喘 并 不 文 持 Redis 开发 环境 ， 只 文 持 使 用 Redis 开发 简单 的 数据 绥 存 
功能 。 所 以 我 们 必须 要 通过 扩展 功能 ， 实 现 Redis 的 编程 文 持 。 为 了 方便 谈 者 学 习 ， 笔 
者 临时 开发 了 相应 的 模块 扩展 及 数据 库 扩 展 ， 下 载 地 址 为 http://beauty-soft.net/book/php_mvc/ 
code/thinkphp_redis.html - 

解压 下 载 后 的 压缩 包 ， 将 得 到 DbRedis.class.php 文件 与 RedisModel.class.php 文件 。 将 
DbRedis.class.php 文件 复制 到 ThinkPHP/Extend/Driver/Db Ha, 将 RedisModel.class.php 文件 
复制 到 ThinkPHP/Extend/Model 目录 。 然 后 在 项 目 配置 文件 中 加 入 Redis 数据 库 连 接 信息 ， 
如 以 下 代 但 所 示 。 


ee 
'REDIS_PORT'=>6379, 

PREDIO AUTHIzszl224bë, 
'REDIS_DB_PREFIX'=>"'! 


读者 根据 实际 环境 填写 即 可 。 通 过 前 面 的 步骤 ， 就 完成 了 在 ThinkPHP 中 进行 Redis H 
发 的 前 期 准备 ， 接 下 来 将 结合 示例 代码 ， 详 细 演 示 Redis 的 CURD 操作 。 

1. 增加 数据 

这 里 的 增加 数据 包括 Redis 五 大 数据 类 型 的 数据 添加 。 由 于 篇 幅 所 限 ， 这 里 不 再 详细 介 
绍 操作 的 实现 原理 ， 而 是 通过 代码 演示 操作 方式 。 如 以 下 代码 所 示 。 
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<?php 
/大 大 


* 


redis 添加 数据 
* Enter description here 
* Qauthor Administrator 
大 
CS 
class AddAction extends Action! 
/** 
zs List eld 
* Enter description bere 
GE 
public function lists (){ 
$Redis=new RedisModel("list11"); 
// 一 次 只 能 推送 一 条 
echo $Redis->add ("ceiba"); 
k 
/** 
* FITRAH 
* Enter description here 
E 
public function String() { 
ŞRedis=new RedisModel();}; 
Şdata=array ( 
EE Ee A 
Tstr2"=>1& Ja", 
"str3"=> "Æ", 
) 7 
echo EE Ehe e ->acde(Scdarta) 7 
/** 
* HASH 类 型 
* Enter description bere 
E 
public function hash (){ 
$Redis=new RedisModel ("user:1"); 
Şdata=array ( 
nel Dn/ EE 
Ve le A 
"field3"=>" 李 明 "， 
); 
// 文 持 批 量 添 加 
echo $Redis->type ("hash")->add ($data); 
} 
/** 
* 集合 类 型 
* Enter description here 
E 
public function sets () { 
sRedis=new RedisModel("sets:1"); 
// 一 次 只 能 推送 一 条 
echo $Redis->type ("sets")->add("ceiba"); 


/** 
* AFRE 
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* Enter description bere 
E 
public function zset (){ 
$Redis=new RedisModel("zset:1"); 
// 文 持 批 量 添加 
sdata=array ( 
// 排 序 => 值 
Eet 
pa 
m12"=>" 李 明 " 
); 
echo $Redis->type("zset")->add ($data); 


} 


2. 查询 数据 
<?php 


Ii redis AWA 
class IndexAction extends Action { 
public function page (){ 
SEnis—>el galay()? 
| 
/** 
* 列表 类 型 ， 默认 类 型 
* Enter description here 
Ge 
public function LILISsStSsS() 1{ 
//dump (C ("REDIS_HOST" ) ) ; 
$Redis=new RedisModel("list1"); 
Şfield=array ( 
"nmae" "age", "pro" 

); 
$data=$Redis->field($field)->select (Oh: 
dump ($data); 

// 获 得 队列 中 的 记录 总 数 
Şcount=$Redis->count (); 
dump ($count); 
| 
/** 
* 字符 串 类 型 
* Enter description here 
E 
public function String() { 
ŞRedis=new RedisModel(); 
// field 表示 每 个 key 名 称 
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SE EE Ee SE e Ee e eh ER EE SE E )=>selece l); 


dump ($rows); 

} 

/** 

* HASH 类 型 

* Enter description here 

Sy 

public function hash (){ 
$Redis=new RedisModel ("h9"); 
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// 默 认 显示 所 有 HASH 字段 ， 可 以 通过 field 连贯 操作 限制 
Şrows=$Redis->type ("hash")->field(array ("field1"))->select(); 
dump (Ş$rows); 
II i RER 
Şcount=$Redis->type ("hash")->count (); 
dump ($count); 
} 
Zë 大 大 
* 集合 类 型 
* Enter description here 
E 
public function sets (){ 
ŞRedis=new RedisModel(); 
Şarr=array ( 
WSSU AY 
); 
$rows=$Redis->type ("sets")—->field ($arr) ->where ("sinterstore")->select () ; // 求 交集 
dump ($rows); 
$rows=$Redis->type ("sets")->field($arr)->where ("sunion")->select ();// 求 并 集 
dump ($rows); 
$rows=$Redis->type ("sets")->field($arr)->where ("sdiff")->select () ;// 求 差 集 
dump ($rows); 
sRedis=new RedisModel ("s3"); 
$rows=$SRedis->type ("sets")->select (); // 返 回 单个 集合 列表 中 的 所 有 成 员 


dump (Ş$rows); 


// 统 计 记录 
$Redis=new RedisModel ("s3"); 
Şcount=$Redis->type ("sets")->count GE 


dump ($count) ; 
} 
/ 大 大 
* 有 序 集合 
* Enter description here 
SC 
public function zset () { 

$Redis=new RedisModel ("z2"); 

// 默 认 显示 0 到 20 
Ee KEE el ele Een EE 
dump ($data) ; 

// 使 用 zRevRange 显示 数据 ， 数 组 第 2 个 参数 为 true 时 显示 排序 号 

EE EE EE (array (vzRevRaRandev, 
crue) Ehel 

dump ($data) ; 

// MRE limit 时 ， 将 统计 所 有 记录 

scount=$Redis->type ("0,1")->count (); 

dump ($count ) ; 


} 


} 
3. 删除 数据 
<?php 
/ 大 大 
* Redis 删除 数据 


* Enter description here 
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aauthor "Administrator 
大 
SE 
class DeleteAction extends Action 
/** 
st 230 
* Enter description here 
E7 
public function lists (){ 
ŞRedis=new RedisModel ("mylist"); 
// 根 据 索 引号 ， 删 除 指定 的 list 元 素 
echo $Redis->where (3)->delete()} 
//1ltrim 区 间 批 量 删 除 ， 保 留 4~5 之 间 的 记录 
echo SRecis EE >wnere (array (VAW, W5W) )=>0elece (er )s 
/ /lpop 单条 顺序 弹出 
EE EE Se EE ee 


/** 
* 字符 串 类 型 
* Enter description here 
E7 
public function string (){ 
$Redis=new RedisModel ii: 
// 和 直接 删除 key， 这 个 方式 适用 于 所 有 数据 类 型 
echo SRecns— Ee Ee Ee Ge SE RE 
i 
/** 
* HASH 类 型 
* Enter description here 
es 
public function hash (){ 
$Redis=new RedisModel ("user:1"); 
/ / MERTE hash 中 的 指定 字段 (field) ,不 支持 批量 删除 
echo $Redis->type ("hash")->where ("field1")->delete();}; 


} 
/** 
* 集合 类 型 
* Enter description here 
SC 
public function sets (){ 
SRedis=new RedisModel ("s1"); 
// 删 除 sets :1 集合 中 名 为 age HJ value 
echo $Redis->type ("sets")->where ("age")->delete(); 
| 
/** 
* 有 序 集合 
* Enter description here 
public function zset (){ 
$Redis=new RedisModel ("z1"); 
// 根 据 集合 元 素 value 进行 删除 
echo $Redis->type("zset")->where("two")->delete(); 


/ (RRIHET S RETK ER, RE 23 之 间 的 记录 
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echo SRedqis->type ("zset")->where (array ("1","4"))->delete ("zremRangeByScore"); 
// 根 据 索 引号 进行 区 间 批 量 删 除 ， 保 留 2~3 之 间 的 记录 
echo SRedqis->type ("zset")->where (array ("1","3"))->delete ("zRemRangeByRank"); 


} 


RS 


在 Redis 中 ， 更 新 数据 与 添加 数据 是 可 以 相互 转换 的 ， 所 以 这 里 不 再 介绍 。 更 多 的 功能 
特性 及 使 用 方法 ， 笔 者 会 进行 更 新 。 本 书 读者 可 在 配套 网 站 中 得 到 后 续 的 文 持 。 


16.4.3 ”数据 分 页 


在 Redis 中 ， 由 于 数据 类 型 之 间 的 结构 不 同 ， 所 以 要 全 部 实现 数据 分 页 ， 是 非常 烦琐 
的 ， 也 是 没 必 要 的 。 这 里 建议 只 对 Zset、List 进行 分 页 。 事 实 上 ， 前 和 面 所 下 载 的 
RedisModel.class.php 模型 只 对 List 数据 类 型 提供 数据 分 页 功能 〈 和 截止 定稿 为 止 )。 在 实现 方 
式 上 与 常规 的 MySQL 数据 分 页 相关 似 。 

为 了 方便 演示 ， 首 先 使 用 phpRedisAdmin 创建 一 个 队列 Key， 并 命名 为 news_list。 在 该 
队列 中 ， 添 加 相应 的 队列 数据 ， 如 网 16-8 所 示 。 


news list 加 x Æ 


Type: list 

m: does not expire 加 
Encoding: ziphst 

Size: 10 items 


中 Add another value 


图 16-8 添加 中 文 队 列 数据 
上 述 步 又 完成 后 ， 束 可 以 在 ThinkPHP 中 进行 查询 并 显示 了 。 如 以 下 代码 所 示 。 


<?php 
class PageAction extends Actiont 
public function index (){ 
import ("ORG.Util.Page"); 


SRedis=new RedisModel ("news_list"); 
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Ş$count=$Redis->count ()， 
// 每 页 数量 
Şpage_size=3; 


ŞPage = new Page ($count, $page_size); 
Sshow = $Page—->show il: 

// 当 前 页 数 

Sie 人 


ER 页 
$first=($page_num-1)*$page_size; 
$list=(($page_num-1)*$page_size+$page_size-l1); 
Sdata=$Redis->limit ($first.','.$list)->select(); 
Şrows=array (); 
foreach ($data as $key=>Şvalue) { 
Şrows [$key] ["title"]=Şvalue; 
} 
Şthis->assign ("page", $show); 
SEN EE De E ; SLOWS) 7 
SE Ee STEE EE 


Ro 


如 上 述 标 粗 代码 所 示 。List 数据 类 型 是 基于 ]Range 实现 的 ， 该 方法 与 MySQL 中 的 limit 
在 处 理 方式 上 有 着 本 质 的 区 别 。 例 如 在 MySQL 中 “limit 5 10” 表 示 从 第 5 条 开始 查询 ， 
次 查询 10 条 ， 边 界 号 为 15; 而 在 Redis 中 ， 则 表示 从 第 5 条 开始 查询 ， 边 界 号 为 10， 共 显 
不 5 条 (10 减 去 5)。 除 此 之 外 ， 其 他 的 调用 代码 与 普通 的 MySQL 分 页 并 无 区 别 。 对 应 的 
index.html 视图 模板 代码 如 下 所 示 。 


<!DOCTYPE html PUBLIC "-—//W3C//DTD XHTML 1.0 Transitional//EN" "nttp://www.w3.org/ 
TR/xhtml11/DTD/xhtml1-transitional.dtd"> 


<html xmlns="http://www.w3.0org/1999/xhtm1l1"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
<title> 数 据 分 页 显示 </title> 
</head> 
<body> 
<div class="main"> 
<div class="content"> 
<ul> 
<volist name="list" id="vo" key="k"> 
<li><!-—-{$vo.title}--></li> 
</volist> 
<ul 
</div> 
<div class="page"> 
<!--{$page}--> 
</div> 
</div> 
</body> 
</html> 


最 终 运行 效果 如 网 16-9 所 示 。 
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BELICE 


é>) U NEE - SS DI 


e Batz olorH mise LI 
e 智能 手机 市 场 国 产品 牌 份额 过 半 利润 却 趟 到 1% 
e 金立 通信 鞋 事 长 刘 立 荣 : 规模 赢 先 机 


10 条 记录 2/4 页 上 一 页 下 一 页 1 2 3 4 


图 16-9 Redis 数据 分 页 效果 


需要 说 明 的 是 ， 由 于 Lit 元 么 并 不 文 持 多 字段 (field)， 所 以 数据 的 储存 及 合 询 将 受到 
局 限 。 例 如 要 储存 新闻 标题 的 同时 ， 还 需要 存储 对 应 的 狐 闻 id， 以 便于 程序 连接 到 相应 的 文 
草 人 全文。 但 显然 通过 上 述 方式 并 不 能 满足 要 求 。 这 里 可 以 使 用 List 元 素 存 储 Json 的 方式 实 
现 ， 或 者 使 用 Hash 数据 类 型 。 


16.5 “小结 


本 章 从 理论 到 实践 ， 详 细 介 绍 了 Redis 的 实战 运用 。 在 全 面 介 绍 redis-cli 终端 操作 命令 
之 后 ， 结 合 PHP 及 PHP MVC 框架 ， 详 细 介 绍 了 phpredis API 的 使 用 。 读 者 在 掌握 了 redis- 
cli 各 种 操作 命令 之 后 ， 无 论 在 哪 种 API 中 操作 Redis， 都 将 变 得 得 心 应 手 ， 而 这 些 内 容 正 是 
本 和 章 所 介绍 的 重点 。 

人 至此， 本 书 关 于 MVC 实战 部 分 的 内 容 残 介绍 完毕 了 ， 下 一 章 将 通过 绽 合 的 实例 ， 进 一 
步 巩 固 本 书 内 容 ， 帮 助 读者 迅速 融入 到 实际 的 项 目 开 发 中 去 。 
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D 17 
开发 论坛 系统 


内 容 提 要 

论坛 ， 也 称 BBS， 是 一 种 较 早 的 互联 网 应 用 。 早 期 的 互联 网 论坛 比较 简单 ， 只 需要 上 传 
文本 及 显示 文本 即 可 ， 随 着 互联 网 的 发 展 ， 现 在 的 论坛 都 是 由 若干 个 功能 模块 组 成 的 ， 其 复 
杂 程 度 不 亚 于 CMS (Content Management System， 内 容 管 理 系统 )。 所 以 怎样 开发 一 个 成 
熟 、 符 合用 户 日 常 使 用 习惯 的 论坛 系统 ， 不 仅 需 要 过 硬 的 技术 ， 还 需要 全 面 且 只 活 的 项 目 组 
织 能 力 。 

本 章 将 学 习 论 坛 系统 的 构建 ， 掌 握 真 实 环境 下 系统 之 间 的 耦合 集成 ， 从 而 为 开发 真正 的 
用 户 产 品 莫 定 基础 。 通 过 本 章 内 容 ， 读 者 将 学 习 到 论坛 系统 的 主要 模块 的 构建 ， 例 如 会 
员 模 块 、 登 录 模 块 、 发 帖 模 块 、 回 复 模 块 等 。 这 些 模 块 都 是 一 个 论坛 系统 中 最 基础 的 应 
用 ， 学 习 完 本 章 内 容 后 ， 读 者 可 以 在 此 基础 上 添加 其 他 模块 ， 从 而 构建 真正 的 全 功能 论 
坛 系统 。 


n) 


了 解 规 划一 个 真实 项 目的 过 程 。 

了 解 使 用 MVC IER ERIS RANAR 
掌握 项 目 整体 界面 的 协调 。 

掌握 将 应 用 整合 到 主流 微 博 的 过 程 。 
巩固 ThinkPHP 标签 扩展 的 应 用 。 

全 面 巩固 本 书 前 耐 所 学 的 内 容 。 
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17.1 开发 前 准备 


论坛 是 互联 网 最 典型 的 应 用 ， 它 与 网 站 留言 板 一 样 ， 是 网 站 与 用 户 进行 在 线 交 互 的 最 主 
要 手段 。 论 坛 系统 是 构建 在 线 论 坛 的 基础 平台 ， 在 国内 外 都 有 比较 成 熟 的 解决 方案 ， 例 如 国 
内 的 PHPWIND、Discuz! 等 。 网 站 开发 者 使 用 这 些 软件 平台 ， 惑 能够 轻易、 迅速 地 创建 功能 
强大 的 在 线 论坛 。 

本 章 将 以 实战 的 方式 ， 全 面 介 绍 论坛 系统 的 开发 。 通 过 本 章 的 学 习 ， 读 者 完全 能 够 
创建 一 个 具有 实际 意义 的 在 线 交 流 平 台 。 当 然 ， 本 章 主 要 的 目的 是 介绍 MVC 开发 论坛 
系统 的 过 程 ， 重 点 是 整合 本 书 前 面 的 内 容 ， 从 而 巩固 所 学 的 知识 。 下 面 首 先 介绍 系统 的 


SEO 


17.1.1 系统 介绍 


虽然 论坛 系统 与 网 站 留言 板 一 样 ， 都 是 网 站 在 线 交 互 的 主要 平台 。 但 论坛 系统 比 留言 板 
更 加 全 面 ， 也 比 留言 板 更 加 复杂 。 在 开发 上 ， 主 要 面 对 的 问题 有 权限 控制 、 板 块 分 类 、 会 员 
管理 、 管 理 员 管 理 、 内 容 管 理 等 。 使 用 MVC 框架 开发 论坛 系统 是 最 合适 不 过 了 ， 因 为 
MVC 的 特性 就 是 前 后 台 分 离 。 论 坛 系统 根据 功能 而 分 ， 是 由 多 个 功能 模块 组 成 的 。 同 时 ， 
MVC 框架 能 够 为 论坛 系统 提供 重要 的 缓存 文 持 、 文 件 处 理 、URL 处 理 每 ， 这 些 特 性 都 是 主 
流 MVC 框架 能 够 完美 文 持 的 技术 。 本 章 继续 以 ThinkPHP 3.x 作为 MVC 框架 ， 详 细 介 绍 论 
坛 系 统 的 开发 过 程 。 

为 了 方便 介绍 ， 这 里 将 系统 分 为 7 大 模块 ， 每 个 模块 均 包 含 徊 干 个 子 模块 ， 我 们 将 在 子 
模块 中 实现 帖子 发 表 、 用 户 注册 、 邮 件 处 理 、 会 员 控 制 等 功能 ， 如 图 17-1 所 示 。 


17.1.2 ”系统 预览 


笔者 已 经 将 本 草 所 介绍 的 论坛 系统 代码 上 传 到 了 配套 网 站 中 。 为 了 方便 读者 学 习 ， 下 面 
将 对 系统 主要 的 功能 模块 进行 预览 ， 以 便 让 读者 对 本 系统 有 一 个 初步 的 认识 。 在 预览 前 ， 首 
先 需 要 将 系统 安 钱 到 本 地 机 器 上 。 

1. 安装 源 代码 

读者 可 以 在 beauty-soft.net/book/php_mvc/code/bbs.html 网 址 下 载 到 本 章 所 介绍 的 论坛 系 
统 源 代码 。 将 下 载 得 到 的 bbs.zip 文件 包 解 压缩 ， 然 后 将 bbs 文件 夹 复制 到 网 站 根 目录 。 打 开 
phpMyadmin 管理 工具 ， 创 建 一 个 数据 库 ， 用 于 存放 论坛 系统 数据 ， 并 将 数据 库 命 名 为 
bbs. 

单 击 “ 导 入 ”链接 ， 并 选择 寻 入 文件 格式 为 “SQL ”， 如 图 17-2 所 示 。 

将 前 面 解压 后 的 databases/bbs.sql 文件 导入 到 bbs 数据 库 中 。 完 成 后 如 图 17-3 
所 示 。 
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项 目 入 口 
文件 


论坛 布局 
论坛 主 版 面 
综合 统计 论坛 数据 


置顶 帖子 列表 
版 块 模块 


普通 帖子 列表 


用 户 控制 面板 
第 三 方 帖 编 辑 器 


处 理 提 交 数 据 
ERME EA 


发 表 帖 子 模块 


回复 帖子 编辑 框 


处 理 回复 内 容 
修改 内 容 


修改 浏览 权限 


帖子 管理 模块 
删除 帖子 


搜索 模块 


图 17-1 系统 模块 


帖子 内 容 模块 


在 导 六 时 脑 本 夺 检 测 到 可 能 需要 花费 很 长 时 间 刚 多 许 中 断 * 尽管 这 会 中 断 事 务 ， 但 在 导 姑 太 交 件 时 是 个 很 好 的 方法 。 


CSN 选项 
en Document Spreadsheet 
8 Sp SQL 兼容 模式 
al 
C) ei 97-2003 FLS Workbook | [wy] 不 要 给 堆 值 使 用 自 增 (AUTO_INCRENERT) 
© Excel 2007 ALSR Workbook T 


(J 


图 17-2 导入 数据 
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目 tbs_adsin 


四 加 加 四 加 四 加 加 回回 四 回回 四 


bbs adain Sroup 
bbs category 
bbs _ reply 

bbs security password 
bbs topic 
bbs_topic_recycle 
bbs_upload_file 
bbs User 

bbs user Eroup 
bbs user inteeral 
bbs weriftication 
bbs weibo binding 
bbs eeibo list 
sph_counter 


图 17-3 系统 数据 表 
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这 些 数据 表 的 作用 将 在 后 面 的 内 容 中 进行 介绍 ， 在 此 只 需要 完成 导入 操作 即 可 。 前 面 的 
步骤 完成 后 ， 接 下 来 只 需要 修改 项 目 配置 文件 (Jbbs/bbs/conf/config.php〉 即 可 。 要 让 系统 下 
第 运行 ， 这 里 需要 修改 的 配置 项 如 下 。 


//' 配 置 项 '=>" 配 置 值 ' 


'DB_TYPE' => 'mysql', // 数据 库 类 型 

Dr => 'localhost', // 服务 器 地 址 

'DB_NAME' => 'bbs, // 数据 库 名 

DEOU ERS => 'root', // 用 户 名 

'DB_PWD' => 'root', // 密码 

'DB_PREFIX' => 'bbs_', // 数据 库 表 前 级 

'TMPL_ PARSE STRING' =>array ( 
' WEBSITE URL '=>"http://tp.localhost", 77 需 要 正确 天 写 
'__BBSPUBLIC _'=>"http://tp.localhost/pbbs/tpl/Public", 
' PROJECT_NAME _'=>"BBS 讨论 区 "， 
'__ PROJECT_ URL _'=>"http://tp.localhost/bbs.php", // 伪 静态 化 绝对 路 径 
'_WEIBO_API_PATH_'=>"http://tp.localhost/bbs", // 微 博 api 接口 地 址 


1, 

上 述 配置 信息 可 根据 实际 的 环境 进行 配置 ， 相 信 遂 过 本 书 前 面 内 容 的 和 学习， 读者 并 不 会 
对 上 述 内 容 感 到 卫生 。 完 成 后 ， 系 统 束 能 够 正常 运行 了 ， 但 全 文 搜索 模块 还 不 能 运行 ， 为 了 
方便 演示 ， 暂 时 不 演示 全 文 搜 索 模 块 ， 这 将 在 后 面 的 内 容 中 进行 介绍 。 

2. 系统 预览 

通过 前 面 的 步 台 ， 系 统 束 处 于 可 运行 状态 了 。 直 接 访问 系统 URL 即 可 ， 首 页 效果 如 
图 17-4 所 示 。 


『 Javait 坛 | 邮箱 : | 用 户 名 / 赋 窒 


【编程 论坛 】 会 员 中 心 
a A zn: | | 


『 Grr 论坛 | 
sam SE CR 
e (éd ges d deeg dgdg i Kiarcebiherb ar 
T mti l T PHP 这 人 坏 J -一 一 - e 
SS AR? Gef Ehe | 登录 | Dam rd LEE 


amwcnngzg, BE 


” 【娱乐 论坛 】 
B El 
【新 闻 论 坛 ] Dër 
veer 
【教育 论坛 】 KE 
BANA 


图 17-4 系统 首页 效果 


读者 可 以 使 用 默认 的 kf@86055.com 账号 登录 系统 (密码 为 1234$6)。 同 时 ， 也 可 以 直 
接 单 击 腾讯 徽 博 账 志 网 标 进行 登录 ， 系 统 会 目 动 完成 账号 的 创建 。 成 功 登 录 后 ， 页 面 上 将 会 
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会 员 登 录 后 ， 上 默认 情况 下 束 允 许 进 行内 容 发 布 了 。 如 图 17-6 所 示 。 


EN EH CH EC 
在 线 论坛 _ 


phpmeed SE per bearia scht me 


发 表 新 主题 
返回 列表 会员 中 心 


TT 

ap, E, zen bgHRRgmäpepaggo BR  — (be 

ANE paat iE E 

AESI |F- b-i H Henna ` e 

会 员 中心 SCC BRAR: KANSAT 


FAR 
和 分 : 27 


HIRT: Ai EATA H 


ERRAN: WHH Eig 


BRRR: 2010118 1228 


ALME 
| 最 新 回复 
帐号 类 型 : SEH "he 
PHASIRI 
登录 帐号 : EIS 
KT 
注册 日 期 : 2013-01-16 122216 
最 后 登录 : 2013-01-16 122216 weg, eegene 
退出 登录 Lari Rm 
图 17-5 会 员 登 录 效 果 图 17-6 发 表 新 主题 


如 果 要 对 现 有 的 主题 帖子 进行 回复 ， 只 需要 在 帖子 的 搬 部 输入 回复 内 容 ， 然 后 提交 即 
可 。 如 图 17-7 所 示 。 


VHS CX UK, 


| an | d'Stagen, Gäns 


图 17-7 回复 主题 
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系统 提供 了 论坛 中 常见 的 帖子 置顶 、 推 荐 、 热 门 等 功能 ， 并 且 在 版 块 中 智能 地 排序 并 以 
图 标 显 示 。 如 图 17-8 所 示 。 


| 图- Soogk CD Bl Op 加- emie: il 


Sp. BER avait 会 员 中心 


Java Kk- SC 
Fis 


PATAR HE0 ESS: S6 [KARS] MAA 


FAHS: SR RSS 


t JavaBean FAJDBCA TRETE RAE MARRA HAMME 17-57 


ERAT: 201301-1 t2216 
t Zendj 近 日 推出 开发 者 云 服务 PHPCloud HATHA HAANEN ? 
BERR: 290 1221 
退出 登录 
最 新 回复 
Saz: [AA] 【推荐 】 
iiaii E 


m RAFAT laddningen] 


图 17-8 论坛 版 块 


前 面 污 示 了 论坛 系统 钊 见 的 会 员 登 录 、 内 容 肥 布 、 内 容 设置 等 功能 ， 还 有 更 多 的 功能 
于 篇 幅 所 限 ， 这 里 不 再 一 一 演示 。 接 下 来 将 结合 代码 ， 详 细 介 绍 系统 的 实现 过 程 。 


17.1.3 ”架构 设计 

这 里 所 说 的 架构 主要 是 指 网 站 的 运行 平台 以 及 数据 库 结 构 。 本 系统 将 使 用 ThinkPHP 
3.0+MySQL 5.1+Sphinx 进行 构建 。 其 中 ThinkPHP 作为 网 站 的 MVC 开 友 框架 ， 主 要 涉及 本 书 
前 面 所 介绍 过 的 内 容 ， 并 重点 介绍 CURD 操作 、 行 为 、 目 定义 标签 、 目 定义 类 库 等 功能 。 

本 系统 共 使 用 了 15 个 数据 表 ， 并 且 能 够 方便 地 进行 扩展 。 这 15 个 数据 表 分 别 为 
bbs_admin、bbs_admin_group、bbs_category、bbs_reply、bbs_security_password、bbs_topic、 
bbs_topic_recycle ~ bbs_upload file ~ bbs user, bb user group ~ bbs_user_lntegral ~ 
bbs_verification、bbs_weibo_binding、bbs_weibo_list、sph_couter。 接 下 来 将 对 数据 表 结 构 进 
行 介绍 。 

1. bbs admin 表 

bbs_admin KEÆ— S SRAN, EHH TRF SEAN MERKEAK A 
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人 码 ， 结 构 如 表 17-1 所 示 。 


表 17-1 be admin 表 
字 段 名 数据 类 型 索 5l 说 H 
admin group id int 管理 员 组 id 

2. bbs admin_group 

论坛 系统 最 重要 的 一 个 概念 束 是 权限 组 概念 。 通 过 权限 组 功能 ， 可 以 将 一 组 管理 员 进 行 
统一 分 配 相 应 的 权限 ， 这 在 大 型 的 论坛 系统 中 是 非常 有 必要 的 。 虽 然 本 章 介 绍 的 论坛 系统 只 
需要 单 管 理 员 《后 台 只 有 一 个 管理 员 )， 但 为 了 系统 更 容易 扩展 ， 这 里 仍然 使 用 通用 的 权限 
组 设计 方案 。bbs_admin_group 数据 表 结 果 如 表 17-2 HZR. 


表 17-2 bbs admin oroup SS 
字 R 名 数据 类 型 SS 5l 说 H 


其 中 permission 字段 用 于 存放 Json 权限 数据 ， 这 些 数据 摘 述 了 该 组 用 户 是 合 拥 有 对 应 的 
操作 权限 ， 例 如 "is_delete_topic":1 表示 访 管 理 组 下 的 管理 员 可 以 删除 主题 。 

3. bbs category 

小 型 的 论坛 系统 ， 如 果 不 考 虑 后 续 扩 展 的 需要 ， 将 论坛 版 块 分 类 数据 存放 在 静态 文件 
中 ， 例 如 XML, Json, HTML 等 ， 是 一 种 理想 的 解决 方案 。 但 是 对 于 需要 经 常 对 版 块 分 类 
数据 进行 修改 或 扩展 的 论坛 ， 将 分 类 数据 存放 于 数据 库 中 ， 就 是 必然 选择 了 。bbs_category 
数据 表 用 于 存放 论坛 版 块 数据 ， 结 构 如 表 17-3 所 示 。 


表 17-3 bbs_ category 表 


RZ 5l 说 W 


数据 类 型 


其 中 parent_id 表示 当前 版 块 的 父 级 id4。 如 果 为 0 则 表示 当前 版 块 为 一 级 分 类 。 
4. bbs_reply 
bbs_reply 表 用 于 存放 主题 的 回复 列表 。 结 构 如 表 17-4 所 示 。 
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表 17-4 bbs_reply 表 


数据 类 型 SS 5l 说 W 


5. bbs security password 
bbs_security_password 表 用 于 存放 用 户 在 注册 会 员 时 可 供 选 择 的 安全 问题 ， 效 果 如 
图 17-9 PTR. 


安全 口令 ; (tele, Dags |w | 用 于 设置 安全 登录 
| 选择 一 个 安全 问题 ， 提 高 登录 安全 性 

RK eren A 全 口令 答案 

上 人 微 博 : 


A t+. |29; I pe 
MFR: Uamesun ies? 


图 17-9 选择 安全 口令 


为 了 方便 扩展 ， 这 里 将 继续 使 用 数据 表 存 放 安 全 提问 。bbs_security_password 表 结 构 如 
K 17-5 所 示 。 


SS 17-5 bbs security_password SS 
字 段 名 数据 类 型 过 一 “号 | 说 H 
id int PRIMARY 主键 


6. bbs topic 

bbs_topic 是 系统 中 比较 重要 的 一 个 数据 表 ， 该 表 用 于 存放 帖子 数据 。 通 常情 况 下 ， 主 题 
表 将 产生 大 量 的 数据 ， 在 设计 时 建议 使 用 数据 表 分 区 技术 ， 以 提 融 数据 表 的 储存 效率 。 
bbs_topic 表 结 构 如 表 17-6 HIZR. 


表 17-6 bbs_topic 表 
字 BR 名 数据 类 型 F 23 说 D 


int PRIMARY 主键 
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Stats tinyint(2) index 状态 


其 中 stas 字段 表示 帖子 的 状态 ， 共 支持 3 种 状态 : -1 表示 帖子 已 经 被 删除 ;0 表示 处 
于 审核 状态 ;1 表示 正常 状态 。 

7. bbs topic_recycle 

管理 员 在 对 帖子 (主题 帖 ) 进 行 删 除 后 ， 系 统 并 不 会 真正 删除 相应 的 数据 ， 而 是 将 stats 
字段 更 新 为 -1。 通 过 这 样 的 设计 ， 就 能 够 安全 地 对 数据 进行 删除 与 恢复 了 。 同 时 ， 为 了 便于 
操作 ， 系 统 会 将 删除 的 帖子 id 记录 到 bbs_repic_recycle 数据 表 中 ， 以 便 记 录 删 除 的 时 间 、 原 
因 、 事 件 等 ， 以 便 恢 复数 据 时 提供 依据 。bbs_topic_recycle 表 结 构 如 表 17-7 PTR o 


子 段 名 数据 类 型 RZ 5l 说 W 


SS 17-7 bbs topic_recycle SS 
字 段 名 数据 类 型 过 “中 说 H 
operation_user 操作 用 户 〈 管 理 员 ) 


8. De _ upload file 

bbs_upload_file 数据 表 用 于 存放 文件 上 传 的 数据 。 用 户 在 上 传 文件 后 ， 除 了 将 上 传 的 文 
Irrel EIEH KAk, EXT Tee Ort, Deem ot, och) Ott 
下 载 等 。bbs_upload_file 表 结 构 如 表 17-8 所 示 。 


表 17-8 bbs_upload file 


SS 5l 说 W 
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数据 类 型 SS 5l 


9. bbs user 
bbs_user 是 一 个 用 户 表 ， 用 于 保存 用 户 的 登录 数据 以 及 联系 资料 等 。 所 以 在 开发 时 需要 
确保 其 安全 可 靠 。bbs_user 表 结 构 如 表 17-9 所 示 。 


说 H 


SS 17-9 bbs user% 


索 


id int PRIMARY 主键 


i 
security_password 无 

安全 问题 答案 
| 
Ei 


last_login_time 最 后 登录 时 间 
user_information medomext 无 | 用 户 联系 资料 (Json 数据 ) 
status tinyint(2) 状态 


其 中 user_information 字段 用 于 存放 用 户 的 联系 资料 (使 用 Json 保存 )， 例 如 联系 QQ 
个 人 主页 等 。 如 以 下 代码 所 示 。 

{"weibo":"", "qq":"102805", "age":"","professional":"", "introduce":"note2 "} 

user_type 表示 用 户 类 型 ， 系 统 文 持 2 种 用 户 类 型 ， 分 别 为 正 第 用 户 和 第 三 方 用 户 。 当 
user_type 为 0 时 表示 正 党 用户， 否则 为 第 三 方 用 户 ， 例 如 来 日 腾 讯 微 博 、 狐 浪 微 博 的 用 户 。 

10. bbs user group 

与 管理 员 一 样 ， 普 通 的 用 户 同 样 也 受用 户 权限 组 的 控制 。bbs_user 表 的 user_group 字段 


index 性 别 
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表示 用 户 对 应 的 用 户 组 。bbs_user_group 用 于 存放 数组 数据 ， 结 构 如 表 17-10 rar, 


表 17-10 bbs_user group 表 
字 BR 名 数据 类 型 SO 说 明 


其 中 permission 字段 用 于 存放 权限 数据 Json 格式 )。 这 些 数 据 描 述 了 该 组 用 户 是 
合 拥有 对 应 的 操作 权限 ， 例 如 "is_browse _title":1 表示 该 用 户 组 下 的 用 户 是 耕 能 够 浏览 
主题 。 

11. bbs user integral 

用 户 在 登录 或 者 发 帖 、 回 帖 时 ， 系 统 将 对 该 用 户 进行 积分 记录 ， 从 而 提高 论坛 的 人 人 气 。 
bbs_user_integral 数据 表 用 于 保存 积分 、 时 间 、 事 件 等 。 结 构 如 表 17-11 所 示 。 


表 17-11 Des user Integral ZS 


数据 类 型 索 5l 


12. bbs verification 

N "eut MRA RE, KALEAREN H ETARE. H rm AER K 
验证 网 址 ， 系 统 将 目 动 完成 用 户 验 证 。 而 在 此 之 前 ， 用 户 处 于 等 竺 验证 状态 。 
bbs_verification 表 用 于 存放 系统 生成 的 验证 码 ， 当 用 户 完 成 验证 后 ， 系 统 将 删除 对 应 的 验证 
码 ， 结 构 如 表 17-12 rar, 


表 17-12 bbs verification 表 
字 段 名 数据 类 型 SS 引 说 ` 


int PRIMARY 主键 
verification char(32) 生成 的 MD5 验证 码 


13. bbs welbo binding 

前 面 提 到 过 ， 系 统 文 持 使 用 微 博 账号 进行 会 员 注 册 、 登 录 等 操作 。 上 此外， 系统 还 文 
持 现 有 的 普通 用 户 将 账号 绑 定 到 多 个 微 博 账 号 。bbs_weibo_binding 数据 表 用 于 存放 微 博 
绑 定 用 户 的 数据 ， 例 如 绑 定 时 间 、 绑 定 的 微 博 账号 、 对 应 的 普通 账 扎 等 ， 结 构 如 表 17-13 
所 示 。 


510 D E 


第 17 章 开发 论坛 系统 


表 17-13 bbs_weibo_binding 表 


数据 类 型 Ro 5l 


weibo_id bbs_weibo_list 表 主 健 id 


14. bbs weibo list 
前 面 提 到 过 ，weibo id 字段 对 应 bbs_weibo list 数据 表 主 键 id。 该 表 用 于 存放 可 供 绑 定 
的 微 博 服务 列表 ， 记 录 了 服务 名 称 、 图 标 、API 地 址 、 状 态 等 数据 ， 结 构 如 表 17-14 所 示 。 


表 17-14 bbs weibo list 表 
字 R 名 数据 类 型 SS 引 说 明 


15. sph counter 
sph_counter 数据 表 用 于 实现 Shinx 实时 全 文 搜索 功能 ， 这 在 本 书 14.4.4 市 已 经 介绍 过 ， 
在 此 不 再 重 述 。 结 构 如 表 17-15 ra, 


表 17-15 sph_counter SS 


数据 类 型 SS 5l 


counter id 统计 id 
上 次 索引 最 大 记录 id 


max doc wd 


17.1.4 系统 部 署 


理解 了 系统 架构 及 数据 结构 ， 接 下 来 束 可 以 看 手 实 现 系统 各 功能 模块 了 。 首 先 需 要 部 车 
项 目 ， 在 网 站 根 目 录 中 创建 一 个 文件 来 ， 访 文件 夹 用 于 存放 ThinkPHP 3.0 框架 文件 以 及 论坛 
系统 源 文件 。 

完成 后 ， 将 二 级 域名 绑 定 到 前 面 创建 的 目录 ， 这 里 绑 定 的 域名 为 http://bbs.localhost。 
接 下 来 创建 论坛 入 口 文件 ， 并 命名 为 bbs.php〈 名 称 可 目 定 义 )， 代 人 码 如 例 程 17-1 
PIR o 
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【 例 程 17-1] bbs/index.php NC OTI, 
<?php 

SEENEN EELER EE 
define (TAPE PATH", H. Zhome/ni: 

define ("APP_NAME", "Home"); 
define ("APP DEBUG", true); 

require_once (THINK_PATH."ThinkPHP.php"); 


直接 访问 http://bbs.localhost/bbs.php， 系 统 将 目 动 创建 bbs 项 目 ， 并 完成 所 有 初始 化 操 
作 。 接 下 来 将 对 项 目 进 行 必要 的 配置 。 打 开 bbs/Conf/config.php 文件 ， 在 该 文件 中 配置 本 系 
统 所 需要 的 配置 数据 。 人 代码 如 例 程 17-2 所 示 。 


【 例 程 17-2] bbs/Conf/config.php 文件 代码 。 
<?php 


return array 人 


//' 配 置 项 '=>" 配 置 值 ' 


(pp TYPE" => 'mysql', // 数据 库 类 型 
ee => 'localhost', // 服务 器 地 址 

'DB_NAME' => 'bbs_demo', // 数据 库 名 

'DB_USER' => 'root', // APZ 

'DB_PWD' => 'root', // 密码 

(pp PEEFIZI => 'bbs_', // 数据 库 表 前 级 


COOKIE PREEIX > 1, Z/Cookie OU 

SESSION PBRbIZIzssTbbe i, 
"SESSION_TRBLE'=>'bbs_session'，// 存 放 session 的 数据 表 名 
E EE 

IO OOKIE RRE Re 60007 

'SHOW_ERROR_MSG' =>true, 

TEXCEPT TON IMPL PttRissiZBublieieszeeptien-btmtt, 
ERROR MESSAGE) = 友和 全 和 7 
'LAYOUT_ON'=>true, 

TOKEN ON falsa, 

'TOKEN_NAME'=>" bbs _", 

DEE EE BULLO CACHEN ~- false, 

URE CASE ITNOENS LIIVE -false 

CURG MODEL = A 

IMEL L DELIM' => '<!—-—{', 

'TMPL_R DELIM' = 

URE CASE INSENSO LIIVEN Crue, 

(MEI, ACTION PEROR IMPL PAIN UPUD IT error nemi, 
CEMPEL ACTION NOES TAPE PATH. POD Succese hemi, 
TME LL EXO EETION FILES -INPE PATI Publi mmp nem, 
TMEL PARSE SIRING: >array| 


' WEBSITE URL _'=>"http://bbs.localhost", 

'__BBSPUBLIC _'=>"http://bbs.localhost/bbs/tpl/Public", 

' PROJECT_NAME _'=>"BBS HEK", 

' PROJECT URL '=>"http://bbs.localhost/bbs .php"，// 伪 静态 化 绝对 路 笃 


t WEIBO API PATH SNEED /DDS Loealbhost/ban, 7 /Ml api lt 
), 
// 网 站 运行 时 设置 ， 可 以 配合 后 台数 据 表 实现 动态 配置 
vC CENAR ILO ON Te -array 
// 用 户 注 册 默 认 状 态 :1 正式 通过 、2 管理 员 审 核 、3 邮件 审核 


"user_reg_stats"=>1, 
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/ /用户 注 册 增 加 积分 
"user_reg_integral"=>10, 

// 用 户 登 录 增 加 积分 
"user_login_integral"=>1, 

/ /用户 发 表 主 题 增加 积分 
"user_post Copie integral"=>20, 
/ /回复 一 个 主题 增加 积分 

Tuser replv Copie jntegralifzszb, 


"user_addtopic_status"=>1, 
"user_addtopic_last_word_count"=>6, 
"user_addreply_status"=>1, 
"user_addreply_last_word_count"=>2, 
"SendMessage"=>array ( 
/ / SMTP 服务 配置 
dé Geh alle 
"host_name"=>"beauty-soft.net", 
"time_out"=>120, 
"bodytype"=>"htm1", 
"user"=>"adminftp", 
"pass"=>"ceiba_535416", 
"mail"=>array ( 
"from"=>"kfQ86055.com", 
"relay_host"=>"beauty-soft.net", 
Hanie Au eaS E 
), 
"FileUpload"=>array ( 
"jmage"=>array ( 
"maxSize"=>1000000, 
EE ("Joons vein one Y Toeg) y 
"savePath"=>APP_PATH. "Uploads/images/", 
"thumbMaxWidth"=>'100,200', 
"thumbMaxHeight"=>'90,160', 
), 
"accessory"=>array( 
"maxSize"=>5000000000, 
EE E e EE om 
"savePath"=>APP_PATH."Uploads/accessory/", 
), 
), 
/ /全文 搜索 配置 
"Sphinx"=>array ( 
pl Ste > GeO 
EE 


1, 
(AE 


EK 


17.2 系统 整体 界面 设计 


成 功 部 普 项 目 后 ， 接 下 来 束 需 要 者 于 实现 系统 的 各 功能 模块 了 。 表 先 需 要 实现 系统 的 布 
局 功能 ， 主 要 包括 系统 的 主体 布局 、 乒 块 布局 。 由 于 版 块 并 不 是 一 成 不 变 的 ， 需 要 根据 帖子 
属性 或 状态 ， 重 能 地 实现 布局 功能 。 下 面 将 对 主要 的 布局 进行 分 析 。 
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17.2.1 布局 方案 


布局 是 一 个 网 站 开发 中 最 基础 也 是 比较 重要 的 准备 工作 ， 相 信 无 论 是 前 人 台 界 面 设计 人 员 
还 是 后 台 编 程 人 员 ， 都 不 乐意 在 杂乱 无 章 的 系统 界面 中 进行 网 站 开发 。 所 以 设计 模块 ， 并 规 
范 化 网 站 主体 皮肤 ， 是 网 站 开发 中 首先 所 需要 面 对 的 问题 。 

在 主流 的 PHP MVC 框架 中 ， 均 提供 了 网 站 布局 功能 。 本 书 6.4.8 市 已 经 对 ThinkPHP 视 
图 布局 功能 进行 了 全 面 介绍 ， 这 里 将 使 用 ThinkPHP 中 的 LAYOUT_ON 功能 实现 论坛 系统 布 
局 。 在 bbs/Tpl 目录 下 创建 layout.html 布局 文件 ， 代 码 如 例 程 17-3 所 示 。 

【 例 程 17-3】 bbs/Tpl/Public/lay.html 代码 。 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" “http://www.w3.org/TR/ 
xhtml11/DTD/xhtml1-strict.dtd"> 

<html xmlns="http://www.w3.0org/1999/xhtm1"> 

<head> 

<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 

<cite> 

<if condition="$pageTitle neq '' "> 

<!—-{$pageTitle}-->- _ PROJECT_NAME 

<else /> 首页 - __PROJECT_NAME _ 

SE 

</title> 


</head> 

<body> 

<div id="header-wrapper"> 
<div id="header"> 


<div id="menu"> 


Se 
<ul> 
<li <eq  name="_GET.id" value=" ">class="current_page_item"</eq>><a 
href=" PROJECT URL EE EE 
<volist name=":NaviGation()" id="vo"> 
<li <e namne= E yalues" Syo, ic >zelass=Veuvunrrenct page 


item"</eq>><a | ede MAD elass="last"><!——1S5vVo. CareDry ELEle} —=— 
></a></1i> 


</volist> 
</ul> 
<1!1-- end 导航 栏 --> 
</div> 
nd enu e 


<div 1d=search"> 


<form method="get" action=" __PROJECT_URL_ "> 
<input type="hidden" name="m" value="search" /> 
<fieldset> 


<input name="words" type="text" id="search-text" onclick="search_ 
cxe (pY value= EE "i> DÉI A K E <else/><!--{$_GET.words}-- 
> 
<label> 
<select name="a" id="search_select" class="search_select"> 
<option value="title"> 搜 索 标 题 </option> 
option value "nal ee | EN option: 
</select> 
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</label> 
<input type="submit" id="search-submit" class="search-submit" value= 
I 
s/Eieldeetz 
</form> 
EE E 
function search 七 Xt () { 
S ("#search-text").attr("value",""); 
} 
</ BCT LOLS 
</div> 
</div> 
</div> 
<!—-— end #header 一 一 > 
<!—- end #header-wrapper 一 一 > 


<div id="1logo"> 

<hl><a href="#"> 在 线 论坛 </a></h1> 

<p><em>PHPMVC 在 线 支 持 论 坛 <a href="http://beauty-soft.net/book/php_mvc/">beauty- 
.net</a></em></p> 

</div> 

< /> 

l== end plogo ==> 

<div id="page"> 


<div id="page-bgtop"> 
<div id="content"> 
{__CONTENT _} 
</div> 
<neq name="m" value="content"> 
<div id="sidebar"> 
<include file="Public:sidebar" /> 
</div> 
</neq> 
<l== ene psidebar ==> 
<div style="clear: both; ">&nbsp;</div> 
</div> 
<l==- end ppage ==> 
</div> 
<div id="footer"> 


<input type="hidden" id="sys_url" name="sys_url" value="<!--tmp-$Think.config.SYS_ 


URL==S>W /> 


<p>Copyright (c) 2012 BeautySoft by <a href="http://beauty-soft.com/book/php_ 


mvc">beauty-soft .com</a>.</p> 


CSS 


</div> 
<!l== end) EGOLEEE ==> 
</body> 
</html> 


ERI a AFH FE AE, KAAR S REH TARAA 
"ZE CH, CR CHE D nl An, RA El CR ERT CHE, bbs/Tpl/Public/Resources 


目录 中 找到 本 系统 所 有 需要 使 用 的 静态 资源 文件 。 由 于 篇 幅 所 限 ， 这 里 不 一 一 列 出 。 
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17.2.2 ”论坛 首页 


上 自负 是 用 户 访 问 论坛 站 和 完 呈 现 的 页 面 。 在 首页 上 通常 会 对 版 块 数据 、 帖 子 主题 数据 、 帖 
子 回 复数 据 、 帖 子 访问 情况 、 用 户 活跃 状态 等 信息 进行 汇总 。 同 时 ， 对 各 大 版 块 、 终 极 版 块 
等 信息 进行 分 区 导航 ， 方 便 用 户 迅 速 找到 感 兴趣 的 内 容 。 对 于 本 系统 而 言 ， 首 页 数据 主要 来 
目 于 bbs_topic、bbs_reply、bbs_category 数据 表 ， 统 计 效 果 如 图 17-10 Bra, 


【编程 论坛 】 


| T cigi l 


『 Javai 人 坛 】 
WERNER 主题 :47 帖 数 :54 
Fd CCCCCCCCCCCCC 只 多 证 会 员 资 客 浏 贤 
『 cat) 『 PHPit 坛 J 
P 主题 :0 帖 数 :0 gœ | 主题 :0 帖 数 :0 


图 17-10 ”首页 数据 统计 效果 


为 了 方便 调用 ， 这 里 将 所 有 版 块 数据 查询 操作 封装 为 CategoryModel 模型 。 访 模型 对 应 
bbs_category 数据 表 。 代 人 码 如 例 程 17-4 所 示 。 


【 例 程 17-4】 bbs/Lib/Model/CategoryModel.class.php 文件 代码 。 
<?php 

class CategoryModel extends Model) 

protected $stats=1; 

/ 大 大 

返回 版 块 数 据 


Eineeneelelsen enue nn 


* 


* Qparam unknown type $id 
sdata=array ("reply"=>false,"reply_data"=>array ()) 
* 说 明 : 
* 1 )reply 为 true 时 ， 将 显示 对 应 主题 的 回复 数据 (默认 10 条 ， 可 通过 reply_data 设置 数量 ) 
EE 参数 值 与 当前 模型 getTopicReply 方法 $data 参数 相同 
e 
public function getCategoryData ($id=0, Ş$data=array()){ 
$id=intval ($id); 
Ştopic=M ("Topic"); 


Şcategory_where["stats"]=$this->stats; 
if ($id>0){ 
// 根 据 id 统计 


Şcategory_where["category_id"]=array ("eq", $id); 
Şwhere_id="topic.category_id=$id"; 

}else{ 
/ /统计 所 有 版 块 数 据 
Şcategory_where["category_id"]=array ("gt", $id); 
Şwhere_id="topic.category_id>$id"; 

} 

if (Şthis->options["where"]){ 
Şcategory_where=array_merge ($this->options[|"where"],$category_where); 

} 

StopicNum=$topic->where (scategory_ where)—>count (); 

Sreply-M ("Reply"); 


516 O B 


第 17 章 开发 论坛 系统 
/7 回帖 数量 


ŞreplyNum=$reply->where (scategory_ where)—>count Il: 
// DI TAE 
ŞtopicAndreplyNum=$topicNum+$replyNum; 
// 当 前 版 块 最 新 帖子 
SEET Kee Cla 
Şlimit=1; 
}else{ 
Slimier= EE ions |S EE SE 
} 
// 默 认 根 据 回复 时 间 排 序 
ii (larray eh Deier TE e Ree besate ee) 
// 根 据 最 新 回复 时 间 , 默认 
ŞDB_PREFIX=C ("DB_PREFIX"); 


swhere="and topic.stats={$this->stats} and top=".intval ($category_where 


propior 
Ssaol=V SELECT COLE. ld ,EGOSLE. cicle cCOo., adel time copie., acb user, 
CODLGC.a Category iC ; 
topic. category_chil id ,topic.top,reply.add time as reply_time 
FROM ~ {S$DB_ PREFIX}topic. as topic, {S$DB_PREFIX}reply as reply where 


(Swhere_ id 
and reply.topic id=reply.id {S$where}) 
SORDERS 
LIMITS 
Şrows=$topic->limit ($limit)->order ("reply time desc,adqd time desc,id 
desc")->query ($sql,true); 
}elsel 
// 根 据 最 新 发 表 时 间 
SEOWSS=HEODLE—>ELElCO( EE ER EE enee LR EE EE ee Ee e 


(Ş$category_where)->limit ($limit)->select ()}; 


} 
// 是 否 显示 回复 数据 
if (Sdatal[l"reply"] Se Srows)t 
foreach (S$rows as $key=>Ş$value){ 
Srows [$key] ["reply"]=$this- 
>getTopicReply ($value["id"], $data["reply_data"]); 
} 


} 
sdatas=array ( 
"topicNum"=>$topicNum, 
"replyNum"=>$replyNum, 
"topicAndreplyNum"=>$topicAndreplyNum, 
"newdata"=>S$rows, 
); 
return S$datas; 
} 
/** 
* 根据 主题 id 获取 应 该 的 主题 回复 
x @param unknown type Sid 
” Scarcasarrtay (Vorder >aoel tine, Wimle “=>10) 
SC 
public function getTopicReply ($id, $data=array ()){ 


E Ek 
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SreplwcchM (fi Beplvni: 
ŞSwhere["id"]=$id; 
Swberelietatahl-—Grthia-z—atataz 
Seele 
Şorder=Ş$data["order"]; 
}else{ 
Şorder="add_time desc"; 
} 
LE (Şcdertal EE EE 国王 | 
Sur Ee Ee Limiet" ] s 
}else{ 
ş$limit=10; 
} 
Şrows=$reply->order ("add_time desc")->limit ($limit)->order ($orde)->where ($where)- 
>select (il: 
return $rows; 
} 
/ 大 大 
* 根据 ID 获取 对 应 版 块 下 的 子 版 块 
* Qparam unknown type $id 
x Qparam unknown type $data 
* Qreturn unknown 
Ge 
public function getCategorySubList ($id,$data=array ()){ 
ŞCategory=M ("Category"); 
Şwhere=array ( 
Wperemer LeWsSH Lel, 
"category_stats"=>1, 


"category_level"=>array ("neq", 0), 


) 7 
Şrows=$Category->where (Ş$where)->order ("category_order desc")->select () ; 
return S$rows; 


2 


完 成 CategoryModel 模型 创建 后 ， 只 需要 在 index 方法 调用 CategoryModel 模型 中 的 
getCategoryData 方法 即 可 获取 所 需要 的 数据 。 代 人 码 如 例 程 序 17-5 所 示 。 
【 例 程 17-5] IndexAction 控制 器 index 动作 代码 。 


public function index (){ 


load ("extend"); 


Şcategory=D ("Category"); 
// 推 荐 主题 
swhere=array ( 
"parent_id"=>0, 
"category_stats"=>1, 
) 7 
Şcategory_list=$category->where ($where)->order ("category_order desc")—>select () ; 
Sthals=>agssion (Yecaregory_ Liget", Scartegory List) 
Şthis->display EE 
| 
对 应 的 bbs/Tpl/Index/index.html 模板 代码 如 下 所 示 。 
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<volist name="category_list" id="vo"> 
<div class="post"> 
elas na span clase adare e e aao aego eae san /> 
<div class="category"> 
<div class="category-list"> 


<volist name=" :CategorySubList ($vo['id'])" id="vos"> 


<div class="category-block"> 


<span class="category-image"> 


<a href="<!-—-{Ş$vos.id|CategoryURL=###}-->" title="" target= 
TESSERE 

<neq name="vos.category_image" value=""> 

<img srce="<!-—-{Ş$vos.category_image}-->" border="0"/> 

<else/> 

<img src=" _BBSPUBLIC /Resources/Image/IE.png" width= 


"90" height="68" border="0"/> 


Jp 省 略 部 分 
</div> 
</volist> 


</div> 
</div> 
</div> 


</volist> 

上 述 步 又 使 用 了 多 个 目 定 义 函 数 ， 例 如 CategoryData, PageURL 等 。 这 些 目 定义 函数 具 
有 非常 重要 的 作用 ， 如 缺少 将 不 能 正常 运行 。 读 者 可 在 配套 源码 包 bbs/Common/common.php 
文件 中 找到 所 有 目 定 义 函 数 代码 ， 这 里 不 再 讲述 。 


17.2.3 ”论坛 版 块 


单 击 首页 中 的 厂 块 图 标 或 版 块 名 称 ， 将 进入 该 版 块 的 终极 栏目 。 但 系统 并 非 只 文 持 两 级 
栏目 分 类 ， 而 是 支持 多 级 栏目 分 类 。 这 里 为 了 方便 讲解 ， 将 终极 栏目 的 上 级 栏目 称 为 大 版 
块 ， 将 终极 栏目 称 为 小 版 块 。 下 面 分 别 介绍 。 

1. 大 版 块 

大 版 块 是 小 版 块 的 上 层 分 类 ， 利 用 大 版 块 分 类 功能 ， 可 以 实现 论坛 的 无 限 级 分 类 (建议 
不 要 超过 3 级 分 类 )。 大 版 块 相当 于 终极 分 类 的 封面 ， 原 则 上 该 版 块 只 用 于 分 类 导航 ， 而 禁 
EFKAR. Ia 17-11 所 示 。 

大 版 块 的 数据 与 首页 所 需要 的 数据 大 致 相同 ， 都 是 取 目 于 CategoryModel 日 定义 模型 中 
的 getCategoryData Jr, Cu Vic, SREL, od "rr 的 控制 器 中 实现 。 这 里 将 
使 用 CategoryAction 控制 右 index 动作 实现 版 块 访问 处 理 。 如 以 下 代码 所 示 。 


public function index (){ 
Load Ee e 


ŞCategory=D ("Category"); ŞobHead=$Category->field("parent_id,id")- 


swaere (Ba (Yict=>9_6BT | Vic]  Yecacegory stars =>) )=->rLmel (y 
if (!ŞṢŞobHead["parent_id"]){ 
// 大 版 块 
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Ei 首页 BBS 讨论 区 
$ e iplocalhost/forum-1-0.html è RW. SF ecr DW: åO A D o 


XEHE pas (KEES TE 


=F a cl d 


/ Dees 
长 1 华 坛 


CT? 


导航 : 首页 >> 编 程 论坛 会 员 中 心 


TH 
[ ti] AEREE 


T teen EA: 

RR EEN A egen 
a 主题 :30 Säi 
e Lgdegdegegegegeg Ku: E ri 


【jawva 论 坛 】 登录 | 哇 图 新 用 户 | | 六 记 密 码 | 


Ee enucggsz, Be 
I Me EE. 
Ge ELE ET 
, 2 E E a E e a 
hii epa EE, A 


Lct] 


Teme 十 
TK ZK: 
dE 


图 17-11 WE Mmmm 


Phi aS Srn lier, aL oy SE CE i | 
Senio eet 


}else{ 
// 小 版 块 
| 
i 


如 上 述 代码 所 示 ， 在 index 动作 中 判断 parent id 字段 是 否 为 0， 如果 是 0 表示 该 版 块 
为 大 版 块 ， 否则 为 小 版 块 。 如 果 是 大 版 块 ， 和 直接 使 用 display 显示 当前 视图 版 块 。 代 人 码 如 例 
fE 17-6 所 示 。 

【 例 程 17-6] bbs/Tpl/Category/index.html 文件 代码 。 


<div class="nav_path"> 导 航 : <span class="path"><!--{:NavPath($ GET['id'])}--></span></div> 
<volist name="list" id="vos"> 
<div class="post"> 
<p class metal <span class "dare kol- ievos category titla) I /span ~r- 
<div class="category"> 
<div class="category-list"> 
<div class="category-block"> 
<span class="category-image"> 
<a href="<!—-{$vos.id|CategoryURL=###}——->" title="" target= 
Ee 
<neq name="vos.category_image" value=""> 
<img srce="<!-—-{Ş$vos.category_image}-->" border="0"/> 
<else/> 
<img src=" BBSPUBLIC /Resources/Image/IE.png" width="90" height 
="68" border="0"/> 
</neq> 
</a> 
</span> 
<span class="category-item"> 
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<p> [ <a href=". SISvos id | CategoryURL Fit) >" < 
vos cac TEE EE > 

<p><em> EW :<!--{$vos.id|CategoryData=###, "topicNum"}--></em> 
<em> 帖 数 :<!--{$Svos.id|CategoryData=###, "topicAndreplyNum"}--></em></p> 

<p> 


<a 人 Tree ere ticle 
人 

人 
SE El EE SE 


</span> 
</div> 
</div> 
</div> 
</div> </ VOLLSES 


2. 小 版 块 

小 版 块 的 处 理 相对 比较 复杂 ， 不 仪 宕 要 处 理 当前 版 块 的 列表 数据 ， 还 需要 处 理 列表 帖子 
状态 。 例 如 是 否 推荐 、 置 项 每。 代码 如 下 所 示 。 

// 小 版 块 

// 置 顶 

Ştop_list=$Category->limit (5)->where (array ("top"=>1))->getCategoryData($_GET["id"]); 

Şthis->assign("top_list",$top_list["newdata"]); 


// 普 通 主题 

import ("ORG.Util.Page"); 

Şcount_array = ŞCategory->where (array ("top"=>0))->getCategoryData ($_GET["id"]);}; 
Şcount=Ş$count_array [|"topicNum"]; 

ŞPage = new Page ($count, 10); 

2 i 

$Page->setConfig('header', (TED. 

// 分 页 样式 定制 


$Page->setConfig("theme", "Ahk HÁ stotalRow% %header% %upPage% %downPage% %firsts 
%prePage% %linkPage% %nextPage% %end%"); 
Şshow = $Page->show();}; 
Şcache_str=md5 ($count .$Page->firstRow.$Page->listRows."_category"); 
Le ($ ($cache str) bel 
şlisr=S (Scache str)? 
}else{ 
Şlist=$Category->limit ($Page->firstRow.','.Ş$Page->listRows)->where (array ("recommend" 
=>0) )->getCategoryData ($_GET["id"],array ("reply"=>false)); 
S (cache sti; S11iscE)} 
} 
ŞcategoryObj=M ("Category"); 
ŞcategoryName=$category0bj->field("category_title")- 
>where (array (Wiel EE E EE ee Er 
sthis->assign("pageTitle", $categoryName["category_title"]); 


Scthis=>assiom (Wlist”, Sligte || ere En e El s 
if (C ("URL_MODEL") ==4) { 
$Sarrl =s array ( 


Wf Eotn (Yeh Ac i atnl? Se EE 
"/\?\&p=\d/i" 

) 7 

Serr2 = array ( 
"forum-\\1-\\3.htm1", 


) 
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Şshow = preg_replace ( Sarrl, Sarr2, S$show ) ; 


} 
Šthis->assign ("navpage", $show); 
Şthis->display ("SubIndex"); 


小 版 块 的 列表 数据 同样 来 目 于 getCategoryData 目 定义 模型 方法 。 但 这 里 需要 详细 地 取 
出 列表 数据 ， 并 且 使 用 Page 扩展 类 实现 分 页 。 

需要 注意 的 是 ， 由 于 论坛 系统 文 持 两 种 URL 模式 : 一 种 是 传统 的 PH 显 式 传 参 方式 ， 表 
现形 式 如 http://bbs.localhost/?m=index&a=category&id=8; 另 一 种 是 为 了 改善 用 户 体验 ， 优 化 搜 
索引 擎 收录 的 目 定义 URLRewriter 模式 ， 表 现形 式 如 http://bbs.localhost/forum-7-1.html， 所 以 
这 里 需要 使 用 preg_replace 正则 函数 对 原 有 的 分 页 链接 形式 进行 改写 。URLRewriter 模式 需要 
配合 服务 器 有 的 URLRewriter 功能 实现 ， 有 具体 可 参阅 配套 源码 包 中 的 .htaccess 文件 。 小 版 块 使 用 
独立 的 视图 模板 ， 方 便 与 大 版 块 进行 区 分 ， 代 人 码 如 例 程 17-7 所 示 。 

【 例 程 17-7】 bbs/Tpl/Category/SubIndex.html VC HEED, 


<div class="nav_path"> $i: <span class="path"><!--{:NavPath($ GET['id'])}--></span></div> 


<div Eelass=" category tirle"><!=- {pagetitle} ==></0iLy> 


<elt name="_GET.p" value="1"> 
<egt name="top_list |count=###" value="1"> 
<div class="post"> 
<p class="meta"><span class="date"> [ika M] </span> </p> 
EE CLASS= ,EMEY E EE 
SE 
<volist name="top_list" id="vo"> 
SLi Claggy LrimkY> 
<em class="top_head_img"><!--{:TopicHeadImage ($vo['id'],PageURL 
(ŝvo['id'])) }=-></em> 
KE cicles El e > nre r=® 
<!--{$vo.id|PageURL=###}-->" ><!--{$vo.title}--></a> 


<span> 
<em> 用 户 :<a href=" APP ?m=user&uid=<!--{S$Svo.add user}—— 
人 时 间 3 和 1 == 
[ee le nn ne H dHg-An, FE) dems 
</span> 
</ LLS 
</volist> 
</div> 
</div> 
</egt> 
</elt> 


<div class="post"> 
w else a el en rE 普通 主题 了 </span><span class= 
"PostTopic"><a href="_ PROJECT URL ?m=posttopic&cid=<!--{$_GET.id}-->"><input type= 
"submit" value=" RR pIE" /></a></span> </p> 
<div class="entry"> 
E 
<div class="v_link_r"> 
<span class "y Link TemA I 人 an 
<span class="v_link_e"> 
<em> 发 表 </em><em> 最 后 回复 </em> 
</span> 
</div> 
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<volist name="1ist" id="vo"> 
<li class="v_ link"> 
<em class="top_head_img"><!--{:TopicHeadImage ($vo['id'],PageURL 
(Svo['id']))}-—></em> 
<a title="<!-—-{$vo.title}-->" href="<!--{$vo.id|PageURL=#ł##} 
——>" SO element le tf le /> [ add:<!-—-{$vo.add_time |date 
aa Ha | 


<span> 

<em class="add_user_img"><img Sreet a ad aer ee 
Data=##ł#,"head_image"}-->" width="25" height="25" /></em> 

<em> 

ga c le a aee aa Daa aer pnicknamert 
ef APP om usersuid <! [vo add user) -0-1 (Svo -add user l uidroData ttit, "user 
nickname" |msubstr=##¢#, 0,3, 'utf-8',false}--> 

</a> 


</em em Leit condition- getLastReplayuser($vol'id']) eg 


en SEI 

<a EI "De ane 
= SW 本 
nickname" |msubstr=##¢#, 0,3, 'utf-8', falsel 一 一 > 

SE 

<else /> 

<a Cirle nl vo add user getLast ke play HH 
nickoame') -Y hre t APP Du er < e vo add user) -l [avo add user lgethast 


和 
</a> 
aa (0 


</span> 
SS 
</volist> 
</div> 
</div> 


</div> 

<div class="navpage"> 
<l== 1 $reg}==> 
<!—-{Ş$navpage}--> 


W ERER B FRIRE EEE, 3 op Dat 数据 列表 为 空 时 ， 
表示 当前 模块 并 不 存在 置 项 帖 ， 所 以 使 用 egt PEER E ERAL. RR 17-12 
所 示 。 


省 fwwWWWwWWWWw BS 时 间 -01 有 月 16 日 16-31 


省 JavaBean 中 使 用 JDBC 方 式 进 行事 务 处 理 “用户 李 开 涌 时 间 -01 月 09917:57 


EE a 


者 zendif 日 推出 开发 者 云 服 务 PHPCIoud ë MAFA 时 间 -08 月 31 日 16-28 


省 fwwWWwWwwWwnw' HAJER 时 间 忆 1 月 16 日 16:31 


P JavaBean 中 使 用 JDBC 方 式 进行 事务 处 理 re 时 间 -01 月 09 昌 17:57 


图 17-12 ”置顶 帖 效果 
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17.3 用户 模块 


用 户 模 块 控制 看 登录 用 户 与 非 登录 用 户 之 间 的 可 操作 权限 ， 将 直接 有 影响 用 户 的 使 用 体 
验 。 所 以 用 户 模 块 的 开发 直接 影响 论坛 平台 的 整体 质量 。 用 户 模 块 的 开 肥 涉及 多 项 拉 术 ， 例 
如 权限 控制 、 安 全 人 处理 、 邮 件 处 理 、 文 件 处 理 等 。 为 了 方便 介绍 ， 接 下 来 只 对 用 户 模 块 中 比 
较 重 要 的 注册 及 登录 子 模块 进行 介绍 ， 这 也 是 整个 用 户 模 块 中 最 重要 的 部 分 。 其 他 的 如 资料 
管理 、 头 像 修 改 、 控 制 面 板 等 该 者 可 以 直接 阅读 本 书 配套 源 代 但 。 


17.3.1 用 户 登 录 


这 里 所 说 的 用 户主 要 是 指 未 登录 的 访客 。 后 面 介绍 的 用 户 则 主要 是 指 登 录 后 的 会 员 。 用 
户 需 要 成 为 会 员 ， 并 拥有 会 员 资格 ， 例 如 发 帖 、 发 表 回 复 、 上 传 图 片 等 就 需要 进行 登录 。 用 
户 在 登录 时 ， 共 有 两 种 方式 。 一 种 是 使 用 论坛 系统 捉 贷 的 用 户 账 喜 ， 邦 外 一 种 则 是 使 用 第 三 
方 服务 帐号， 例如 腾讯 微 傅 账 写 、 新 浪 微 博 账号 等 。 

无 论 使 用 哪 种 方式 登录 论坛 系统 ， 都 必须 经 过 系统 的 验证 模块 。 为 了 方便 介绍 ， 下 面 首 
和 完 介 绍 使 用 系统 账 写 登录 的 过 程 。 

自 完 用 户 在 登录 表单 中 输入 登录 邮箱 及 密码 ， 并 提交 表单 ， 后 合 验证 模块 将 对 用 户 的 登 
录 数 据 进行 验证 ， 如 果 验 证 和 失败， 提示 数 据 错 误 或 者 用 户 不 存在 ;， 用户 可 以 重新 输入 正确 的 
账号 或 密码 尝试 重复 登录 ， 或 者 单 击 注册 连接 进入 账号 注册 模块 ， 过 程 如 图 17-13 所 示 。 


成 功 


记录 Session 


登录 成 功 


图 17-13 ”用户 登录 流程 


用 户 在 登录 过 程 中 ， 如 打出 现 寞 各 将 提交 给 信息 处 理 模 块 进行 处 理 ， 访 页 面 允 许 用户 返 
四 “上 一 页 ””“ 版 块 列表 ”以 及 “会 员 注 册 ” 页 面 。 而 当 用 户 登 录 成 功 后 ， 信 息 处 理 页 面 将 
直接 导航 到 用 户 控 制 面 板 。 这 里 将 使 用 ThinkPHP 提供 的 error 及 success 方法 实现 消息 处 
理 ， 对 应 的 模板 位 于 bbs\Tpl\Public 目录 (由 配置 文件 TMPL_ACTION_ERROR 及 TMPL_ 
ACTION SUCCESS 项 决定 模板 路 径 )。 用 户 表 单 页 面 由 公共 模板 sidebar.html 渔 染 ， 代 码 如 
例 程 17-8 所 示 。 
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【 例 程 17-8】 bbs/Tpl/Public/sidebar.html 文件 代码 。 
<div id="User-Login"> 
<div id="minicard"></div> 
<if condition="$login eq 1 "> 
LeCriprt CwvpescfDtest/iauascriptnz 
$ ("#minicard") .load(™ PROJECT URL "+"?m=user&a=userminicard"); 
SE SE 
<else /> 
<form method="post" id="userloginform" name="userloginform" onsubmit= 
"return userlogin(this)" action=" PROJECT URL ?m=user&a=userlogin"> 
eps i 8: <input name="user email” onclick-"clname();" type="text" 
dësser email” value EI Ee d'Ent 
</P> 
ps EI, <input type="password" name="password" id="password" /></p> 
<p><span> 验证; </span><span><input type="text" size="6" class="Verification" 


id="Verification" /></span><span><img src="__APP_ /index/verify" onclick="show (this);" 
/></span></p> 

sp input Cypre ENEE i vale te 3 a 
和 
hrer Y PBOTECT URL — 2mtUSerbässgtetpasewerdit Earoetssh blankt ll das a 

<p class="wei icon><span> 使 用 其 他 账号 登录 : </span> 

Span a d Tao Link eire wo aa rer Ror URL om 
userga weibologinswid ln 00 lks G K/a- span 

Spang id SinaLink" title "HHFA AE hirer € PROJECT. 


URL ?m=user&a=weibologin&gwid=2"> 使 用 新 浪 微 博 账户 登录 </a></span></p> 


</form> 


EE 省 略 JavaScript 部 分 代码 


GE 
</div> 


最 终 的 界面 效果 如 图 17-14 所 示 。 Rëm 
用 户 单 击 登录 表单 后 ， 将 提交 到 UserAction 控制 器 下 的 “全 全 
userlogin 动作 进行 处 理 。 代 码 如 下 所 示 。 邮箱 : | 用 户 名 /邮箱 


EE ab 

* 处 理 用 户 登 录 数 据 — EE 
2 验证 : ASENA 
public function userlogin(){ 

TS wiag 
$User=D ("User") ; 登录 | rant bkg] 
e 使 用 其 它 帐 呈 登录 : BS 
SEET Eelere 

$this->ajaxReturn ($_POST, "验证 人 码 出 错 ", 0); 图 17-14 ”会 员 登 录 表单 
} "a es 
SE (la=7za=270=9_=])r (s [a=2A=20=9 =])/W,S POST 
["user_email"])){ 
$sthis->ajaxReturn ($_POST, "邮箱 出 错 ", 0); 
} 
了 


Sthis->ajaxReturn ($_POST, "密码 最 少 4 位 数 "，0) ; 
} 
swhere=array ( 
"user email"=>h($ POST["user email"]), 
"password"=>md5 ($_POST["password"]) 
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); 
Ş$rows=$User->where ($where)->find(); 
if (Şrows[|"status"]==1){ 
/ /登录 成 功 
session("user id",md5 (Srows["id"| .$s POST["user email"]|)); 
session ("password",md5($_POST["password"])); 
cookie ("user_email",$_POST["user_email"]); 
// 增 加 积分 
Şconfig=C ("SCENARIO_CONFIG"); 
人 
// 增 加 登录 次 数 
ŞUser->addUserLoginNumber ($rows["id"]); 
$TMPL PARSE STRING=C ("TMPL PARSE STRING" ) ; 
SUurl=$TMPL PARSE STRING[" PROJECT URL "|]."?m=user&a=userminicard"; 
$this >ajaxReturn(array (user >l, or url), ne Rs 
}elseif ($rows["stats"]==2)1{ 
sthis -ajaxReturn (S POST, T Ae- OWS user emailia do ls JU mi: 
}elseif ($rows["stats"]==3){ 
Sthis->ajaxReturn(S_POST," 你 的 用 户 资 格 管理 员 正 在 审核 中 ") ; 
}else{ 
$this->ajaxReturn ($_POST, "登录 失败 ， 请 检 奏 用 户 名 或 邮箱 ".$User->getLastSsSd1l1 ()，0) ; 


} 


} 
上 述 校 验 过 程 使 用 了 目 定 义 UserModel 模型 ， 访 模型 代码 如 例 程 17-9 所 示 。 
【 例 程 17-9】 bbs/Lib/Model/UserModel.class.php 文件 代码 。 
<?php 
class UserModel extends Model { 
protected $_validate=array ( 
array oser emari Semai AE Emaar AE, 
arrac user onea a emaa l E O aa aE 
array (oa Ona Hreouiren, A an na 20n1. 
arrav ('repassword', 'password',;, (BEA EAR 0, confirm), 
Ilarra user pnicknamen, it, a Oe Di, 
array verii Ee n, 
array (user nickname., user nickname IEMA Ta, 


1: 


/大 大 
* 判断 当前 用 户 是 否 已 经 登录 
Se 


public function UserLogin(){ 
$sUserIīId=session ("user_id"); 
ŞUser=M ("User"); 
// 用 户 ida+email 等 于 session 中 存放 的 user_id 
ŞuRows=$User->where (array ("user_email"=>cookie ("user_email")))->field("id,user_ 
EE 
if (!$sUserId==md5 (Ş$uRows["id"].Ş$uRows[|"user_email"])){ 
session ("user_id",null); 
cookie ("user_email",null); 
recura 0z 
} 
ŞsUserPassword=session ("password"); 
Şwhere=array ( 
Ee KEE | WLW ] ， 
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"password"=>$sUserPassword, 
); 
Ş$rows=$User->field("status")->where ($where)->find(); 
return intval ($rows["status"]); 
} 
/** 
x 根据 sessiq 获取 当前 登录 用 户 数 据 
* Qparam unknown type Suser_ sessid 
GE 


public function UserData ($user_sessid){ 


ssUserId=session("user id")， 
SsUser=M("User"); 
if (Şthis->UserLogin()){ 


ŞuRows=$User->where (array ("user_email"=>cookie ("user_email")))->find(); 
if ($sUserId==md5 ($uRows[|"id"].Ş$uRows[|"user_email"])){ 
ŞuRows["integral"]=Ş$this->sumUserIntegral (Ş$uRows["id"]); 


return $uRows; 


}else{ 


Log: :write ("获取 当前 登录 用 户 数 据 时 产生 了 错误 : ".SUser->getLastSql()); 
return "获取 数据 出 错 ， 请 联系 管理 员 "， 


} 
}else{ 


return "请 登录 后 再 操作 "; 


} 
/** 
* 获取 用 户 积分 列表 
x @param unknown type Sid 
x Qparam unknown type S$field 


SE 
public function getUserIintegral ($id, $field="")f{ 

Şuser_integral=M("UserIntegral"); 

em ve e e | 
Şrows=Ş$user_integral->where (array ("user"=>$id))->select () ， 
return S$rows; 

}else{ 
Şrows=$user_integral->field($field)->where (array ("user"=>$id))->selct(); 
return $rows[$field]; 

} 

} 
/** 


* 统 可 用 户 发 表 的 主题 数量 
* Qparam int S$id 
= Qreturn int 
7 
public function getUserTopicNumber ($id)t 
Ştopic=M ("Topic"); 
ŞtopicNum=$topic->where (array ("add_user"=>$id, "stats"=>1))->count (); 
return $topicNum; 
} 


/** 


* 统计 用 户 回 复 帖子 数量 
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* Qparam unknown type $id 
* Qreturn unknown 
CH 
public function getUserReplyNumber ($id) { 
SreplwcchM (fi Beplvni: 
ŞreplyNum=$reply->where (array ("add_user"=>$id, "stats"=>1))->count (); 
return Ş$replyNum; 
} 
/** 
人 
* Qparam unknown type $id 
GE 
public function sumUserIntegral ($id)t 
Şuser_integral=M("UserIntegral"); 
Şsum=Ş$user_integral->where (array ("user"=>$id))->sum("integral"); 
return $sum; 
} 
/** 
* 增加 用 户 登 录 次 数 
大 
gd 
public function addUserLoginNumber ($user) { 
ŞUser=M ("User"); 
load ("extend"); 
swhere=array ("id"=>Ş$user); 
ŞUser—->where ($where)->save (array ("last_login_ip"=>get_client_ip())); 
ŞSnum=$User->where ($where)->setInc("login_number",1); 


return $num; 


P 


17.3.2 用户 注 册 


当 访 问 用 户 单 击 “ 注 册 新 用 户 ” 链 接 后 ， 系 统 将 引导 用 户 进 入 会 员 注册 页 面 。 注 册 流 程 
如 图 17-15 所 示 。 


注册 新 用 户 


邮件 系 乡 


Ee 
表单 检验 “| -通过 保存 成 功 


处 理 失败 


音 息 处 理 


邮件 处 理 完 毕 
广 册 成 功 


失败 


图 17-15 注册 新 用 户 
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这 里 使 用 register 动作 作为 注册 表单 页 面 ， 表单 验证 将 使 用 UserModel DC VR 
$_validate 学 段 完 成 ;邮件 处 理 使 用 SendMessage 扩展 类 完成 〈 有 具体 可 参考 本 书 11.2.7 节 )。 
下 面 将 首先 实现 register 动作 ， 代 码 如 下 所 示 。 


/** 
SE 
e? 
public function register (){ 
if (Şthis->userlogin){ 
Sthis->success ("你 已 经 成 功 登 录 ")， 


} 
ssecurityPassword=M("SecurityPassword"); 
ssecurity_list=$securityPassword->order("sort desc,id desc")—>select () ; 
SENLS—>asSsSlon (VeaeurLey Liget”, Ssecuricy lisc)? 
Şthis->display ();}; 

} 

对 应 的 register.html 视图 模板 代码 如 例 程 17-10 所 示 。 


【 例 程 17-10] bbs/Tpl/User/register.html 文件 代码 。 


<div class="post"> 
<form name="forml" method="post" action=" _APP ?m=user&a=registerpost"> 
<p class="meta"><span class="date"> 【用 户 注册 】</span> </p> 
<ul style="font-size:14px"> 提 示 : 如 果 你 已 经 有 账号 ， 可 以 直接 在 右边 登录 窗口 中 进行 


> 
<div class="main"> 
< te Nl e 
<legend> 基 础 选项 </legend> 
<ul> 
<e ea e: <input type test! name user email size- 
"30"><em> ÑH 5AA CVH) </em></1i> 
<li>- Š a SI. <input type="password" name="password" 
size="30"><em> 蜜 但 不 能 少 于 4 位 数 〈 必 填 ) </em></li> 
<li> m % I: <input type="password" name="repassword" 


size="30"m><em> 重 复 一 次 密码 </em></1i> TH <input type "tert" 
name="user_nickname" size="30"™><em> 用 户 呢 称 ， 支 持 中 文 ( 必 填 ) </em></li> 
</ul> 
</fieldset> 


<fieldset olass VEDert nr style “display nonet title va 
<legend- ANA a E) < / Legends 
<ul> 
e ELE 
<select name="security_password" > 
<option values ns A] TEHA, REKLE 


性 </option> 
<volist name="security_list" id="vo"> 
<option value="<!-—-{$vo.password}-=-->"><!-—- 
{$vo.password}--></option> 
</volist> 


</select> 
em HI TO a a dems d LI 
liste input type "tekt" size "30" name security 


_answer"><em> 安 全 口令 答案 </em></1i> 


< NL ve eet name "weibo value vt 


EE EE 
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See nu EE tesi name euer size TOn h 

ele o0 <input type “text! name tgga" size "IOT -</i> 

EA E na ee Ee aa roe eona Size 
ss LUH zec/L1z 


<textarea name="introduce" class="introduce" rows 


SEENEN 


</ul> 
</fieldset> 
<p><input type="checkbox" name="che 1" onchange="ShowExpert (this); 
"><em> 高 级 选项 </em></p> 


<div class="sub"> 
<input type="submit" value=" 同 意 条 款 并 注册 "><em> 阅 读 条 球 </em> 
</div> 
</div> 
</form> 
</div> 


//…… 省 略 部 份 代码 
最 终 的 界面 效 来 如 图 17-16 PR. 


€ |2 iplocalhostbbs.php?m=user&a=register @ a- ZE -<riKP Xr- È- AD ó ht- z-a 


图 17-16 用 户 注 册页 面 效 果 
用 户 单 击 “ 同 意 条 于 并 注册 ”按钮 ， 将 提交 到 registerpost 动作 处 理 。 代 人 码 如 下 所 示 。 
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/** 
* 处 理 注册 数据 
CH 
public function registerpost (){ 
ŞUser=D ("User"); 
Loaec Ae el > 
Sdata=$User->create(); 
Şuser_information =array ( 
"weibo"=>h ($_POST["weibo"]), 
EE Lee BE ee ($_POST Yagati); 
"oal -h(S Eeer 
"professional"=>h ($_POST["professional"]), 
"introduce"=>h($_POST["introduce"]) 
IR 
if ($data) { 
Şuser_information=json_encode ($user_information); 
sdatal"user information"]=$user information; 
sdatal[l"security answer"]=md5 ($data["security_answer"]); 
Scarcal "reg Cimenl-ztmeii: 
Scarcal V srtarcs™ ]=Sconir ig | user reg stars] p 
Şconfig=C ("SCENARIO_CONFIG"); 
if (ŞUser->add ($data) ){ 
if (Şconfig["user_reg_stats"]==3){ 
// 发 送 邮 件 验 证 信 ， 在 实际 开发 中 可 结合 消息 队列 《如 HTTPSQS) 进行 处 理 ， 以 提高 性 能 
Ee raLlse) 
Şconfig=C ("SCENARIO_CONFIG"); 
import ("@.ORG.SendMessage"); 
$verification=md5 (time().$ POST["user email"]|); 
smailInfo=array ( 
ooreet irgo TEn a, 
"verification" =>$verification, 
"time"=>time () 
) 7 
SE Ee Ee (Wma LW, Snel mee) E: 
// 邮 件 内 容 
Şbody=$this->fetch("Public:Mail"); 
SendMessage::init ($config["SendMessage"]); 
if (SendMessage::SendMail ("欢迎 注册 丽 物 论坛 会 员 "， S$body, $_POST["user 
SE 
// 将 验证 码 保存 到 数据 库 
$verification=M ("Verification"); 
$data["verification"]=$verification; 
$data["add time"]=time(); 
if (!$verification->add ($data) ){ 
Log: :write ("在 保存 邮件 验证 码 时 产生 了 错误 : ".S$verification-> 
getLastsql()); 
throw new Exception ("验证 人 码 插 入 失败 ， 请 联系 管理 员 手 动 审核 "); 


Lelsel 
Log: :write ("在 给 注册 会 员 发 送 验 证 邮件 时 发 生 了 错误 : " .SendMessage: :S$qepbug) ; 


} 
/ /增加 积分 
AddUserIntegral ($rows ["id"], $config["user zeg integral"], "新 用 户 注 册 积 分 ")， 
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sthis->success (" 用 户 注册 成 功 "，_ APP ."?m=user&a=login"); 


Lelsel 
Log: :write ("用 户 注册 出 错 : ".SUser->getLastSql()); 
throw new Exception ("数据 插入 失败 ， 请 联系 管理 员 "); 
} 
}else{ 
Şthis->error ($User->getError()); 


} 


} 

上 述 代码 中 ， 使 用 了 SendMessage 目 定 义 扩展 类 ， 并 且 将 邮件 系统 配置 信息 配置 到 了 
config.php 中 。 所 有 自 定义 扩展 类 库 均 保存 到 bbs/Lib/ORG 目录 ， 所 以 SendMessage 扩展 类 
的 真实 路 径 为 bbyLib/ORG/SendMessage.class.php。 由 于 篇 幅 所 限 ， 这 里 不 再 对 该 类 进行 介 
绍 ， 具 体 使 用 方式 可 阅读 本 书 前 面 讲述 过 的 内 容 。 


17.3.3 ”使 用 微 博 账号 登录 


使 用 现 有 的 第 三 方 账 亏 进行 登录 ， 和 是 有 效 提高 用 户 体验 的 重要 方式 。 由 于 主流 SNS 系 
统 的 普及 ， 使 用 这 些 账 号 登录 可 以 免 去 用 户 注 册 账 号 的 兵 烦 ， 用 户 只 需要 成 功 登 录 第 三 方 服 
务 ， 就 可 以 实现 在 当前 系统 中 登录 ， 并 拥有 正常 会 员 的 权限 。 

当前 ， 主 流 的 SNS 对 外 提供 服务 都 是 基于 OAuth 2.x 协议 的 ， 所 以 无 论 安全 性 ， 还 是 便 
捷 性 ， 都 是 开发 SNS 应 用 的 可 靠 技 术 。 作 为 PHP 程序 员 而 言 ， 只 需要 下 载 官方 提供 的 API 
即 可 实现 所 有 基于 OAuth 认证 的 功能 或 服务 ， 例 如 登录 、 获 取 数 据 、 上 传 数 据 等 。 这 里 只 需 
要 使 用 登录 服务 即 可 。 

国内 的 主流 SNS 应 用 都 提供 了 OAuth 服务 ， 并 且 都 提供 有 完善 的 API， 开 发 人 员 下 载 
后 只 需要 修改 少量 数据 ， 即 可 实现 第 三 方 账号 登录 服务 。 接 下 来 将 以 腾讯 微 博 为 例 ， 介 绍 腾 
讯 微 博 的 OAuth 登录 服务 。 

1. 下 载 和 配置 微 博 接口 

OAuth 是 一 余 开 放 的 授权 标准 ， 任 何 上 商都 可 以 使 用 。PHP 本 号 就 能 够 文 持 OAuth 标 
准 ， 所 以 调用 任何 基于 OAuth 的 第 三 方 服务 并 不 珊 要 额外 的 扩展 。 但 为 了 方便 使 用 ， 这 里 将 
直接 使 用 官方 提供 的 SD 玉 。 读 者 可 以 在 http:Wwiki.open.t.qq.coryindex.php 网 站 中 下 载 到 本 章 
所 需要 使 用 的 API 文件 。 

同时 ， 本 书 配套 源码 中 同样 提供 了 相应 的 API， 文 件 位 于 bbs/otherlogin/TencentWeibo H 
录 中 。 打 开 SDK 目录 下 的 Config.php 配置 文件 ， 根 据 实际 情况 填写 。 代 码 如 下 。 


<?php 

HA5 B ETA appid 

Selient 16 = "801065573" 7 

EE HEH] appkey 

$client_secret = '94a5c459cb55826ceeb6c34bb50a4c80'; 

// 调 试 模式 

S$Sdebug = false; 

官方 提供 的 appid X appkey 能 够 实现 基本 的 功能 ， 读 者 也 可 以 正式 申请 专 有 的 appid 及 
appkey。 由 于 本 间 内 容 只 需要 登录 服务 ， 所 以 使 用 官方 提供 的 束 已 足够 。 

为 了 方便 介绍 ， 接 下 来 并 非 将 腾讯 微 博 SDK 整合 到 论坛 系统 中 ， 而 是 使 用 JavaScript 处 
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理 用 户 登 录 成 功 数 据 。 只 需要 简单 修改 demo.php 文件 即 可 。 代 人 码 如 下 所 示 。 


LE ($ SESSION|[ "EE accsss token” | || ($_SESSION['t _ openid'] && Ş_SESSION['t_openkey'])) 
a 
// 获 取 用 户 信 息 
$r = Tencent: :api('user/info'); 
// 随 机 生成 表单 令 牌 


$api_token=rand(100, 12000) .time () ; 
$5_SESSION["api token"]=md5 ($api_ Cokeni: 
sdata=json decode ($r,true); 
?> 
<html xmlns="http://www.w3.0org/1999/xhtml1"> 
<head> 
<meta http-equiv="content-type" content="text/html; charset=utf-8" /> 
<cite> 
</title> 
<sCripr type= "text/javascript "> 
setTimeout ("document .weiboform.submit ()",2) 
EE 
</head> 
<body> 
<form name="weiboform" method="post" action="http://bbs.localhost/bbs .php?m=user&a= 
receiveWeiboPost"> 


<input type="hidden" name="source" value="1"> 


<input type="hidden" name="nickname" value="<?php echo $data['data'] ['nick'];?>"> 
<input type="hidden" name="username" value="<?php echo $data['data'] ['name'];?>"> 
<input type="hidden" name="head_image" value="<?php echo $data['data'] ['head'];?>"> 


<input type="hidden" name="api_token" value="<?php echo $api_token; ?>"> 
<input type="hidden" name="data" value="<?php echo S$r;?>"> 

</form> 

</body> 

</html> 


<?php 


HEN 

如 上 述 代码 所 示 ， 为 了 能 够 让 第 三 方 服务 返回 的 登录 数据 能 够 提交 (POST) 到 论坛 系 
统 中 ， 使 用 了 随机 令 牌 机 制 ， 并 登记 到 Session。 通 过 这 些 步 又 束 能 够 防止 非法 的 数据 来 
源 ， 确 保 数 据 来 日 于 微 博 服 务 。 

2. 处 理 登 录 数 据 

微 清 返回 的 数据 了 最终 将 提交 到 UserAction 控制 器 下 的 receiveWeiboPost 动作 。 不 管 是 
哪 一 个 第 三 方 服 务 ， 需 要 提交 的 数据 字段 都 包括 nickname、name、Source、head_imasge、 
opt Token, 

在 receiveWeiboPost 动作 中 ， 处 理 的 流程 主要 包括 数据 校 验 、 用 户 检 测 、 数 据 处 理 、 
Session 登记 等 。 如 图 17-17 所 示 。 

图 中 ， 整 个 登录 过 程 分 为 2 条 主线 : 当 微 博 账 号 已 经 存在 系统 时 ， 例 如 执行 过 绑 定 操 
作 、 或 者 非 第 一 次 执行 微 博 登录 ， 数 据 库 中 将 会 保存 该 用 户 信 息 ， 此 时 系统 只 需要 在 
Session 登记 即 可 ， 以 便 确 认 登 录 状 态 ， 而 当 数 据 库 中 并 不 存在 该 微 博 用 户 时 ， 为 了 提高 用 
户 体验 ， 系 统 将 上 自动 注册 该 用 户 ， 但 这 一 过 程 是 后 台 目 动 完 成 的 ， 并 不 会 给 用 户 任 何 提示 ， 
用 户 最 终 看 到 的 是 登录 成 功 信 息 〈 即 返回 用 户 中 心 首 页 )。 整 个 流程 都 将 在 receiveWeiboPost 
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动作 中 完成 ， 代 码 如 下 所 示 。 


第 三 方 账号 登录 


数据 来 源 检 验 


进入 新 注册 流程 


图 17-17 mb mt MII 


public function receiveWeiboPost () { 
LE (wel ($_POST Yapi_coken” ]) =$ SESSIONI Yapi oren i)i 
//dump(S_SESSTON); 
$this->error ("HRA") ; 
} 
ŞUser=D ("User"); 
Şuser_data=$User->UserData (session ("user_id")); 
Şweibo=M ("WeiboList"); 
load ("extend"); 
Şweibo_rows=Ş$weibo->where (array ("id"=>$_POST[|"source"],"status"=>1))->find(); 
Şsource=Ş$weibo_rows[|"id"]; 
Şconfig=C ("SCENARIO CONFIG"); 
Ssweibo _ binding=M ("WeiboBinding"); 
Suser name=empty($_ POST["username"|])?s POST["nickname"] :$_ POST["username"]; 
if ($source && Suser name) { 
Susername=$Sweibo rows["mark"]." H. Guser name; 
Şrows=$User->where (array ("username"=>S$Susername, "status"=>1))->find(); 
Şweibo_binding_rows=$weibo_binding- 
>where (array ("weibo_id"=>Ş$source,"weibo_username"=>Ş$user_name))->find(); 
if (Şweibo_binding_rows){ 
Şrows=uidToData ($weibo_binding_rows[|"user"],"",true); 
} 
if ((ŞṢrows[|"user_type"]==Ş$source && !session ("binding")) | | Sweibo binding rows) { 
/ /保存 登录 数据 


session("user id",md5 ($rows["id"] .$rows["user email"])); 


session("password", srows["password"]); 

cookie ("user_email", $rows["user_email"]); 

// 增 加 积分 

AddUserIntegral ($rows["id"], $configl"user Login _ integraln],n 使 用 微 博 登 录 
增加 积分 ") ; 

// 增 加 登录 次 数 

ŞUser->addUserLoginNumber ($rows["id"]); 

jump ("?m=user"); 

}elseif (session("binding")){ 
/ AA K AE E 
SS 省 略 部 分 代码 
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// D aigEI 
sdata=array ( 
"username"=>$username, 


Was emani Y= ww, 


"password"=>md5 (rand (100, 


"reg_time"=>time(), 
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1000) .time()), 


"user nickname"=>empty($_ POST["nickname"])?Suser name:$ POST 


["nickname"], 


"last_login_time"=>time(), 


slast login to¥=>get_cClisnt 100) p 


Tatatush-—zt, 
"user_type"=>Ş$source 
) 7 
if ($UseLc->adqdq(S$aqatal) ) { 
有 


Srows=$User->where (array ("username"=>$username, "status"=>1))->find(); 


session ("user_id",md5 ($rows["id"] .$rows["user email"])); 
session("password", $rows["password"]); 
Cookie("user email", $rows["user email"]); 


AddUserIntegral ($rows["id"],S$config["user reg integral"]," 使 用 微 博 


账号 注册 积分 ")， 
jump ("?m=user"); 
Lelsel 


Log: :write(" 用 户 在 使 用 微 博 登 录 时 产生 了 严重 错误 : ".$User->getLastsql()); 
throw new Exception ("操作 失败 ， 请 联系 管理 员 ")，; 


} 


}else{ 


Log: :write(" 微 博 账 户 不 存在 :" .S$Sweibo->getLastSql 1() ) ; 


局 


17.3.4 将 现 有 账号 绑 定 到 微 博 


普通 的 用 户 在 登录 会 员 中 心 后 ， 可 以 将 现 有 的 账 
写 绑 定 到 微 博 上 。 这 样 ， 用 户 在 登录 时 可 以 单 击 微 链 
接 进 行 登录 ， 也 可 以 使 用 内 置 的 账号 进行 登录 。 当 
然 ， 这 个 过 程 是 可 逆 的 ， 用 户 可 以 随时 进行 解除 绑 定 
操作 。 如 图 17-18 所 示 。 

图 中 ， 判 读 用 户 是 否 已 经 绑 定 ， 主 要 根据 bbs_ 
weibo_binding AJER AWW RK. ARKIT ME O 
定 用 户 的 相关 数据 ， 例 如 绑 定 微 博 服务 商 、 绑 定 微 博 
账号 、 绑 定时 间 等 。 如 有 条 用户 需要 解除 绑 定 关系 ， 只 
需要 将 该 表 中 对 应 的 记录 删除 即 可 。 

整个 绑 定 流程 关键 之 处 在 用 户 数据 检验 步骤 ， 系 
统 需 要 对 SDK 返回 的 微 博 账号 等 数据 进行 校 验 。 例 
如 判读 该 微 博 账号 是 否 已 经 存在 ， 服 务 商 是 否 合法 


微 博 授 权 


用 户 数据 校 验 


X 
H 


通过 


查询 绑 定 状态 


未 绑 定 


已 绑 定 普 | 进入 解 绑 定 流 程 


删除 


图 17-18 


微 博 绑 定 流程 图 
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zeg 


等 ， 这 些 操作 同样 是 在 receiveWeiboPost 动作 中 完成 ， 代 人 码 如 下 所 示 。 


public function receiveWeiboPost () { 
if (md5($_ POST["api token"])!=$_ SESSION["api token™])t{ 
{L dump (8 SESSTON); 
$this->error ("HIERA") ; 


} 

ŞUser=D ("User"); 

Şuser_data=$User->UserData (session ("user_id")); 

Şweibo=M ("WeiboList"); 

load ("extend"); 

Şweibo_rows=Ş$weibo->where (array ("id"=>$_POST[|"source"],"status"=>1))->find(); 

Şsource=Ş$weibo_rows[|"id"]; 

Şconfig=C ("SCENARIO CONFIG"); 

Ssweibo binding=M ("WeiboBinding"); 

Suser name=empty($_ POST["username"|])?$ POST["nickname"] :$_ POST["username"]; 

if ($source && Suser name) { 
Susername=$Sweibo rows["mark"]." ".S$Suser name; 
Şrows=$User->where (array ("username"=>S$Susername, "status"=>1))->find(); 
Şweibo_binding_rows=$weibo_binding- 

>where (array ("weibo_id"=>Ş$source,"weibo_username"=>Ş$user_name))->find(); 
if (Şweibo_binding_rows){ 
Şrows=uidToData (S$weibo binding rows["user"],"",true); 


} 


if ((S$Srows["user type"|]==$source && !session ("binding")) | | Sweibo binding rows) { 
/ /保存 登录 数据 
session("user id",md5 ($rows["id"] .$rows["user email"])); 
session("password", Ş$rows["password"]); 
cookie ("user_email", $rows["user email"]); 
/ /增加 积分 


AddUserIntegral ($rows["id"],s$config["user login integral"], "使 用 微 博 登录 
增加 积分 ")， 

/ /增加 登录 次 数 

ŞUser->addUserLoginNumber ($rows["id"]); 


jump ("?m=user"); 


}elseif (session ("binding")){ 
// 账 号 绑 定 操作 


$user_id=$user_data["id"]; $weibo=$weibo_binding->where (array ("weibo_ 
id"=>$source, "user"=>$user_id, "weibo_username"=>$user_name) )->count (); 
if ($rows["user type"] !=$source && !Ş$weibo){ 
sdata=array ( 
Yuger => $user Lal, 
"binding_time"=>time(), 
"weibo_ id"=>$source, 
"weibo username"=>s$user_name 
); 
if (S$Sweibo binding->add ($data))t 
segs Lon (Volnei me" noll1)e 
Sthis->success (" 账 号 已 经 成 功 绑 定 ") ; 
Lelsel 
Log: :write ("用 户 在 进行 账号 绑 定 时 ， 产 生 了 一 个 错误 : ".Sweibo_binding- 
>getLastsql()); 
$this->error ("账号 绑 定 失败 ， 请 联系 管理 员 ")， 


}else { 
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session("binding",null); 
$this->error ("该 账号 已 经 被 使 用 ， 不 能 再 绑 定 ")， 


} 


Lelsel 
// D sit H 
, 9 Doä2n 
| 
}else{ 


Log: :write(" 微 博 账户 不 存在 :".S$Sweibo->gqetLastSdl () ) ; 
$this->error (" 登 录 超 时 ， 请 重新 探 作 ") ; 
} 


} 
解除 绑 定 操作 将 在 removeBinding 动作 中 完成 ， 代 码 如 下 所 示 。 


public function removeBinding (){ 


if (Şthis->userlogin==1){ 
Şweibo_binding=M ("WeiboBinding"); 
SUSer=D El EE 
$user_data=$User->UserData (session ("user_id")); 
suser id=$user datal"id"]; 
Şweibo=$weibo_binding->where (array ("weibo_id"=>$_GET["weibo_id"],"user"=>Ş$user_ 
LCN) -> Cotrim EE 
if (Şweibo){ 
if ($weibo_binding—->where (array ("weibo_id"=>$_GET ["weibo_id"] , "user"=> 
$user_id))->delete()){ 
Sthis->success ("已 经 成 功 解 除 绑 定 "); 
Lelsel 
Log: :write(" 用 户 在 解除 微 博 账号 绑 定 时 产生 了 一 个 错误 : ".S$weibo_pinding- 


>getLastsql()); 
$this->error (" 解 除 绑 定 失败 ， 请 联系 管理 员 " ) ; 
} 
}else{ 
Sthis->error ("不 存在 绑 定 关 系 ")， 
} 


}else{ 
$Sthnis->error ("请 登录 后 再 操作 ")， 
} 


17.4 ”发 表 帖 子 模块 


发 表 帖 子 相 对 比较 简单 ， 只 需要 使 用 ThinkPHP 提供 的 CURD 操作 即 可 完成 。 本 书 前 面 
的 内 容 中 已 经 对 ThinkPHP 内 置 的 编辑 器 进行 过 详细 介绍 ， 读 者 可 以 直接 引入 到 发 表 帖 页 
面 。 但 这 里 为 了 帮助 读者 更 好 地 掌握 ThinkPHP 自 定义 标签 功能 (使 用 方式 可 参考 本 书 6.3.7 
节 )， 将 使 用 百度 编辑 器 作为 发 帖 编辑 器 。 


17.4.1 整合 百度 编辑 器 
百度 编辑 器 简称 UEditor， 是 一 套 遵 循 W3C 标准 的 强大 在 线 编辑 器 。UEditor 强大 之 处 
不 仅 在 功能 上 ， 而 且 还 体现 在 部 署 方式 上 。UEditor 使 用 JavaScript Fr, Cat 
O Een 
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UEditor 能 够 运行 在 任何 Web 环境 ， 在 使 用 时 只 需要 引入 UEditor 脚本 类 库 即 可 。UEditor 的 
详细 介绍 与 帮助 信息 读者 可 以 在 http://ueditor.baidu.com/website/ 官 方 网 站 中 获取 。 接 下 来 将 
直接 进入 UEditor 的 实战 开发 。 
本 书 6.3.7 节 中 ， 己 经 详细 介绍 过 ThinkPHP 标签 库 的 使 用 。 使 用 自 定 义 标 签 类 库 的 好 
处 是 可 以 在 类 库 中 定义 自 ， 然后 在 视图 模板 中 以 XML 的 方式 引入 这 些 自 定 标 义 
签 ， 这 不 仅 能 够 提高 开发 效率 ， 同 时 也 让 代码 维护 变 得 容易 。 接 下 来 将 详细 介绍 通过 目 定 义 
标签 实现 和 百度 编辑 器 的 引用 。 
定义 标签 库 
日 定 义 标签 可 以 全 部 定义 在 一 个 类 库 中 ， 但 为 了 便于 讲解 ， 这 里 将 使 用 一 个 独立 的 标签 
库 实现 编辑 器 目 定 义 标 签 。 所 有 上 自 定 义 标签 类 库 必须 保存 到 ThinkPHP/Extend/Driver/TagLib 
目录 ， 这 里 创建 的 标签 库 为 Bbs。 并 且 创 建 一 个 目 定 义 标签 ， 命 名 为 editor。 在 editor 标签 中 
实现 UEditor 的 配置 及 初始 化 ， 代 码 如 例 程 17-11 所 示 。 


【 例 程 17-11] ThinkPHP/Extend/Driver/TagLib/TagLibBbs.class.php 文件 代码 。 
<?php 
class TagLibBbs extends TagLib{ 
// 标签 定义 
protected Stags = array ( 
// 标签 定义 : attr 属性 列表 close 是 否 闭 合 (0 或 者 1 默认 1) alias 标签 别名 level H 


SEA 
EE EE 

); 

/** 

NE RD OT TT RT EEE EDE E 

* ueditor 标签 解析 插入 可 视 化 编辑 器 

vl bbs- -editor 1d "textContent" editurlsthttp://urln theme "default" config “editor 
_config" word="true" element="true" width="" height="" cleartdata="false"></bbs:editor> 

EE EE EP Te S a EE EE 

SE 

EE EE EE 

* Qparam string $attr 标签 属性 

EE 

nz Breturn stringlvoid 

EE EE EE eR EE 

Ka 


public function _editor($attr, $content) { 
Şweb_config=C ("SCENARIO_CONFIG"); 
Stad = $this->parseXmlAttr ($attr, 'editor'); 


Sie = lemoty (Scag " iet] 2 EGG | 3 Yrexcarcea" y} 

stheme = !empty ($tag['theme']) ? Ş$tag['theme'] : 'default'; 

Slang = lemorcy (Seaol leng") 2 Stagi lang’ 8 "zn-=-em"} 
Şconfig=!empty (Ş$tag['config']) ? Ş$tag['config'] : 'editor_config'; 
ŞwordCount=!empty ($tag['word']) ?Stad['word'] : 0; 
ŞelementPathEnabled=!empty ($tag['element']) ? Ştag['element'] : 0; 
ŞminFrameHeight=!empty (Ş$tag['height']) ? Ş$tag['height'] : 0; 
Şwidth=!empty ($tag['width']) ? $tag['width'] : 0; 
ʻSauctoClearinictialConctent=:!l empcy (Sa  clearcdarca”l) ?Scaclvelearoaa ll s 0p 
Surl=sgrtag | "vedievwel Y] 7 

ŞimageUrl=_APP_ ."?m=UploadFile&a=uploadimage"; 
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SsUser=D ("User"); 


$user_data=$User->UserData (session ("user_id")); 
$fileUrl= APP _."?m=UploadFile&a=upload&user=".Şuser_data['id']."&token=".Ştoken; 
ŞimageManagerUrl=__APP_ ."?m=UploadFile&a=index"; 
ŞgetMovieUrl=__APP_ ."?m=UploadFile&a=getmovieurl"; 
sweburl=C ("TMPL_PARSE_STRING"); 
Ş$weburl=$weburl['_ WEBSITE URL "Il: 
Saase ke = " 
<script type='text/javascript'> 
el 

window.UEDITOR_HOME_URL = '{$url1}'; 
E 
</script> 


<script type='text/javascript' src='{$url}{$config}.js'></script> 
<script type='text/javascript' src='{$url}editor_all.js'></script> 
<script type='text/plain' id='".$id."' name='".$id."' style='width:".$width."px'> 
"$content." 
</script> 
<script type='text/javascript'> 
UE.getEditor('".$id."', { 
theme: '".$theme."', 
Lang: rn. Slang "mr, 
minFrameHeight : $minFrameHeight, 
wordCount : $wordCount, 
elementPathEnabled:$elementPathEnabled, 
autoClearinitialContent:$autoClearinitialContent, 


imageUrl1: '".$imageUrl."', 

imagePath:'', 

EaeU IU Sfileuri. ae 

filePath:'', 

imageManagerUrl1: '".$imageManagerUrl."', 
imageManagerPath:'".S$weburl."', 
wordImageUrl:'".$imageUrl."', 


wordImagePath:'', 
getMovieUrl: '".$getMovieUrl."', 
maxInputCount:2 


DE 


</script> 


"e 
£ 


return $parsesStr; 
} 
} 


Po 


如 上 述 标 粗 代码 所 示 束 是 初始 化 UEditor 所 需要 的 代码 。 关 于 UEditor 的 配置 及 使 用 ， 
由 于 不 是 本 书 所 介绍 的 范畴 ， 所 以 这 里 不 再 讲述 。 如 果 有 必要 ， 读 者 可 以 从 官方 帮助 文档 中 
获得 帮助 。 定 义 editor 标签 后 ， 束 可 以 在 视图 模板 中 和 直接 使 用 了 。 

假设 下 载 的 百度 编辑 器 源 代码 保存 到 bbs/Tpl/Public/Resources/ueditor HK, ARA editor 
标签 的 引用 方式 如 以 下 代码 所 示 。 


<bbs:editor id="textContent" editurl="/bbs/Tpl/Public/Resources/ueditor" cleartdata 
="false"” config="editor config"> 初 始 化 显示 内 容 </bbs:editor> 
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由 于 editor 标签 位 于 Bbs 标签 库 ， 所 以 还 需要 手动 引入 该 标签 库 ， 代 但 如 下 。 


<taglib name='Bbs' /> 
效果 如 图 17-19 所 示 。 


ei D :三 * te E5 


ss 全 BI UWA- ie d 


-CENCE 


E 


图 17-19 UEditor 效果 


编辑 器 的 样式 及 外 观 是 由 配置 文件 来 决定 的 。http://ueditor.baidu.com/website/ipanel/panel.html 
将 生成 后 的 配置 文件 下 载 后 ， 保 存 到 UEditor 


编辑 
网 址 能 够 生成 所 有 UEditor 支持 的 样式 及 功 
所 在 目录 ， 然 后 在 调用 editor 标签 时 ， 指 定 配置 文件 即 可 ， 如 以 下 代码 所 示 
editurl=" BBSPUBLIC /Resources/ueditor/" config="editor 


an S 
<bbs :edqitor id="textContent" 
width="650" height="220" > 


config_2" 
回复 内 容 
</bbs:editor> 


效果 如 图 17-20 所 示 。 


ZMEHe ER KX UW 


sl: 


图 17-20 定制 UEditor 外 观 


17.4.2 上 传 图 片 
下 载 UEditor 后 ， 不 仅 得 到 编辑 器 所 有 源 代码 ， 官 方 还 提供 了 一 系列 后 台 处 理 接 口 ， 所 
能 用 于 演示 ， 但 在 实 


有 文件 位 于 ueditor/php 目录 。 需 要 注意 的 是 ， 官 方 提供 的 PHP 代码 只 
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际 应 用 开发 中 还 要 处 理 诸多 问题 ， 例 如 限制 上 传 权 限 、 上 传 类 型 、 上 传 大 小 等 。 显 然 使 用 官 
方 的 API 是 不 能 满足 开发 需要 的 ， 所 以 这 里 将 直接 整合 到 ThinkPHP 框架 中 ， 利 用 框架 提供 
的 上 传 功能 ， 不 仅 能 够 有 效 地 处 理 上 传 文件 ， 还 可 以 实现 上 传 权 限 控 制 。 前 面 在 初始 化 
UEditor 时 ， 已 经 对 上 传 路 径 进 行 了 初始 化 ， 代 人 码 如 下 。 


Ee D EE E E 
这 里 的 $imageUrl 变量 的 值 如 下 。 
ŞimageUrl=_APP_ ."?m=UploadFile&a=uploadimage"; 


上 述 代码 表示 使 用 UploadFile 控制 器 下 的 uploadimage 动作 完成 图 片上 传 。uploadimage 
动作 代码 如 下 所 示 。 


ET 
0 


public function uploadimage () { 
C("LAYOUT ON", false); 
if ($this->userlogin==1){ 
Şuser=D ("User"); 
Şuser_data=$user->UserData (session ("user_id")); 
if ( lIUserPermission(suser datal"id"], "is_upload_images")){ 
echo json encode (array ("state"=>" 权 限 出 错 ") ); 
exit (0) $ 
} 
import ("ORG.Net.UploadFile"); 
sconfigs=C ("SCENARIO CONFIG"); 
Şconfig=$configs['FileUpload'] ['image']; 
supload = new UploadFile(); 
Şupload->maxSize = Sconfig["maxSize"] ; 
Şupload->allowExts = Sconfig['allowExts"]; 
Şupload->savePath = $config["savePath"]; 
Şupload->saveRule=time(); 
Şupload->thumb=true; Şupload->thumbMaxWidth=empty ($config["thumbMaxWidth"])? 
100: $config["thumbMaxWidth"]; supload->thumbMaxHeight=empty ($config["thumbMaxHeight"])? 
90:5config["thumbMaxHeight"]; 
Şupload->thumbPath=$config["savePath"] ."thumb/"; 
Şupload->uploadReplace=true; 


if (!$upload->upload()) { 
echo json_encode (array ("state"=>Ş$upload->getErrorMsg())); 
exit (0)? 

}else{ 


Şinfo = Ş$Şupload->getUploadFileInfo(); 
EE EE lO] s 
Şweburl=C ("TMPL_PARSE_STRING"); 
Ş$weburl=$weburl[|'__WEBSITE_URL_']; 
sdata=array ( 
Wirel U= eur eerst 
"title"=>$infol"name"], 
"original"=>Ş$info["name"], 
Wee Be E 
) 7 
echo json_encode ($data); 
} 
}else{ 
echo json_encode (array ("state"=>" 请 登录 后 再 操作 ") ); 
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} 


} 
经 过 目 定 义 后 的 上 传 醒 块 ， 不 仅 可 以 实现 对 上 传 权限 的 控制 ， 还 可 以 方便 地 利用 
ThinkPHP 强大 的 上 传 功能 ， 完 成 对 图 片 截图 、 打 水 印 等 钊 见 操作 。 


17.4.3 ”管理 图 片 


UEditor 编辑 帮 还 近 供 了 在 线 管理 图 片 的 功能 ， 用 户 可 以 通过 在 线 浏览 ， 选 择 所 需要 的 
图 片 ， 完 成 编辑 堪 的 图 卢 插 入 。 同 样 ， 在 初始 化 时 也 需要 将 该 模块 整合 到 论坛 系统 中 ， 代 码 


如 下 。 
imageManagerUrl:'".Ş$imageManagerUrl."', 
$iageManagerUr 变量 值 如 下 。 
ŞimageManagerUrl=__APP_ ."?m=UploadFile&a=index"; 


UploadFile 控制 器 下 的 index 动作 代码 如 下 所 示 。 

public function index (){ 
header ("Content-Type: text/html; charset=utf-8"); 
error_reporting( E ERROR | E_WARNING ) ， 
sconfigs=C ("SCENARIO_CONFIG"); 


Şconfig=$configs[|'FileUpload'] ['image']; 
spath = $config["savePath"]."thumb"; 
//echo $path; 
Şaction = htmlspecialchars( empty($ POST[ "action" ])?Ş$_GET['action']:Ş$_POST 
De ee LONN; 
ŞUploadFile=D ("UploadFile"); 
LE ( Şactcion == Yger ) í 
Şfiles = SUploadFile->getfiles( $path ); 
if ( 'Şfiles ) return; 
reort (S rLLleS, SORT STRING) > 
E = 1; 
roreach ( Srilles as SElle ) 4 
SS8EE = $Srile . Yve separare uey 


b 


echo $str; 
} 
} 


运行 效果 如 图 17-21 所 示 。 


SI? EA 


图 17-21 图 片 在 线 管理 效果 
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17.4.4 上 传 附 件 


附件 一 般 是 指 比较 大 型 的 压缩 包 文 件 。 在 上 传 方式 上 ， 一 般 使 用 Flash 技术 ， 以 便 实 现 
实时 上 传 进度 显示 效果 ， 提 高 用 户 体验 。UEditor 使 用 流行 的 swfupload 组 件 实 现 附 件 上 传 ， 
在 后 台 配 置 上 ， 与 普通 的 几 记 上传 并 没有 多 大 区 列 ， 只 需要 退回 以 下 Json 数据 格式 即 可 。 


{ 


eet EE // 保 存 后 的 文件 路 径 
CEET // 文 件 摘 述 

Same en Wb. be et // 原 始 文 件 名 

ey : ' SUCCESS'" // 上 传 状态 ， 成 功 时 返回 SUCCESS 


} 


在 前 面 的 初始 化 操作 中 ， 附 件 上 传 配置 如 以 下 代码 所 示 。 


te E i ke uri Anu 
$fileUrl EUn LA FARI ZR o 
ŞfileUrl=__APP_ ."?m=UploadFile&a=upload&user=".Ş$user_data['id']."&token=".Ş$token; 


由 于 Flash DLRA AN m Session 数据 ， 所 以 这 里 使 用 token SENS HE, Wy 
止 未 授 权 的 用 户 进 行 上 传 ， 造 成 安全 隐患 。upload 动作 代码 如 下 所 示 。 
/** 
* 上 传 附 件 
ES 
public function upload(){ 


SToSLE=D (VTOPLEY ) $ 

I/S Ttoken=$ Tope => hedik roken EE | Yeici] $ _ Gar mser] $ Onl tokent); 

$token=1; 

iE ($rcoken) i 
C("LAYOUT ON", false); 
Şuser=D ("User"); 
$user_data=$user->UserData (session ("user_id")); 
import ("ORG.Net.UploadFile"); 
Şconfigs=C("SCENARIO_CONFIG"); 
Şconfig=$configs['FileUpload']['accessory']; 
supload = new UploadFile(); 


Şupload->maxSize = Sconfig["maxSize"] 
Şupload->allowExts = Sconfig['allowExts"]; 
Şupload->savePath = $config["savePath"]; 


Şupload->uploadReplace=true; 
Şupload->saveRule=time(); 


if (!$upload->upload()) { 
echo json_encode (array ("state"=>"ERROR", "info"=>$upload->getErrorMsg () ) ); 
}else{ 


Şupload_file=D("UploadFile"); 

Şinfo = Ş$upload->getUploadFileInfo(); 

EE EE Ell 

sdata=array ( 
"strkey"=>md5 (Ş$info["savename"]), 
"user"=>intval(s$ GET['user'|]), 
"adqdq time"=>time(), 
"type"=>$info[l"extension"], 
"file path"=>sinfo[l"savepath"] .$info["savename"], 
Wcopie iclY=>incval ($ CET eire” I); 
reply EE? 
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); 
if (!S$Supload file->add(sdata))t 
Log: :write ("用 户 在 上 传 附件 时 产生 了 一 个 错误 : ".$upload->getLastSql ()); 
} 
sdata=array ( 
"url"=> APP_ _."?m=download&cid=".md5 ($info["savename"]), 
"title"=>$infol"name"], 
"original"=>$info["name"], 
Dotare EE E 
) 7 
echo json_encode ($data); 
} 
}else { 
echo json_encode (array ("state"=>"ERROR", "info"™=>" 权 限 出 错 ") A 


EE 


最 终 运行 效果 如 图 17-22 所 示 。 


附件 (Ni 


( eu JI mn | 


图 17-22 ”附件 上 传 效 果 


17.4.5 ”数据 提交 处 理 


通过 表面 的 学 习 ， 相 信 读 者 已 经 掌握 发 表 帖 子 的 页 面 设 计 了 。 用 户 在 提交 数据 时 ， 将 提 
交 到 PosttopicAction 控制 器 下 的 add 动作 中 处 理 。 代 人 码 如 下 所 示 。 


/** 
* 发 表 / 修 改 主题 
a 


public function add(){ 
if (Şthis->userlogin==1){ 
SE SE (V TODLGE) 
Şconfig=C ("SCENARIO_CONFIG"); 
ee 
if (ccStrLen(trim($ POST["textContent"]))<s$config["user addtopic last_word_count"] 
| | dE El Ee (Erim(s el EE ) <Sconr GEERT Ee ee E BEE 
word_count"] 
Il 
$this->error ("标题 或 内 容 必 须 多 于 " .Sconfig["user_addtopic_last_word_count"]." 
a 
} 
SUser=D (YUser Y) y 
$user_data=$User->UserData (session ("user_id")); 
sdata=array ( 
wrie let=>$_POST EEN See i], 
"content "=>$_POST["textContent"], 
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"add user"=>S$Suser datal"id"], 

dE EE EE 
"browse_user_group"=>$_POST[|"browse_user_group"], 
Wackl ipy=>ger Client 19) 7 

WEE | user Sooodole scacus™ l, 


E; 


if ($_POST[|"ac"]=="update" && !empty($_GET["id"])){ 
// 更 新 数据 
sdatal[l"last _ update time"]=time(); 
i OST lol et le E el CH 
Sesseom(W om "no 


$this->error ("非法 的 数据 来 源 ") ; 
} 
$row=$Topic->where (array ("id"=>intval($_ GET["id"])))->save ($data); 
$message=" 更 新 成 功 "; 
}else{ 
// 插 入 数据 
sdatal[l"adqd time"]=time (); 
Şrow=$Topic->add ($data); 
Smessage WD TE Ni; 
} 
session(" token_ ",null); 
if (Srow)t 
Şthis->assign("url",session("CategoryUrl")); 
Şthis->success ($message); 
}else{ 
有 
echo $Topic->getLastsql(); 
$this->error ("操作 失败 ， 请 联系 管理 员 ")，; 
} 


}else{ 


$Sthnis->error ("请 登录 后 再 操作 ")， 


17.5 帖子 内 容 模块 


用 户 单 击 版 块 中 的 标题 链接 后 ， 将 进入 帖子 正文 。 帖 子 正 文 将 直接 呈现 帖子 的 内 容 ， 包 
括 内 容 全 文 、 权 限 处 理 、 回 复 列表 等 。 下 和 面 分 别 介 绍 。 
17.5.1 帖子 正文 

帖子 正文 是 使 用 ContentAction 控制 喜 进 行 处 理 的 。 代 但 如 下 所 示 。 


public function index (){ 


SOBLe=D( ToD 


Şrows=$topic->where (array ("id"=>$_GET["cid"],"stats"=>1))->find(); 
ŞUser=D ("User"); 

Şuser_data=$User->UserData (session ("user_id")); 
Şconfig=C("TMPL_PARSE_STRING"); 

if (!Şrows){ 
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$this->error ("帖子 不 存在 "); 
} 
if (Srows["browse user _ group"]==1)1 
if ($rows["add user"] !=$user datal"id"] && Ş$this->userlogin!=1){ 
$this->error ("该 内 容 只 允许 会 员 浏 览 "); 


} 
ue rn el ee en a 
SC EE wis DOronse Comcanc e 
$this->error ("权限 错误 !")， 
} 
SEODDLE cache SEr= Vode cache Y.S GiT erc] 
SContenmc=S (STO00 Le Cacnea SEE) A 
if (empty ($toppic_cache_str)){ 
S (listr Cahe str, Seoncenc); 
} 


SEnls— e EE e e Ee let, Srowsl Y ecicleY]) g 
Şthis->assign("content", $rows); 
Şuser_data=uidToData ($rows["add_user"]); 
// 回 复 

TEE 省 略 部 分 代码 


EE ee EH (e 
Sthis->assign("user" $user data); 
Şthis->display ();}; 


相信 通过 本 书 前 面 内 容 的 学 上 述 代 码 读 者 不 会 感觉 生 涩 ， 因 为 这 些 内 容 都 是 
ThinkPHP 中 基础 的 CURD 操作 。 要 注意 的 是 权限 判断 。 例 如 判断 当前 用 户 组 是 合共 
备 浏 览 内 容 的 权限 ， 以 及 是 否 强制 为 注册 会 员 才 能 浏览 。 总 而 言 之 ， 权 限 欣 erën 
统 中 最 需要 讶 慎 面 对 的 问题 ， 一 旦 处 理 不 好 将 直接 涉及 系统 的 安全 ， 权 限 控 制 不 能 通过 从 单 
的 示例 演示 ， 需 要 读者 在 中 实 的 环境 中 根据 情况 进行 处 理 。 

与 小 版 块 设计 一 样 ， 正 文 页 面 同样 支持 目 定 义 UrlRewriter 模式 ， 但 ThinkPHP 的 分 页 扩 
展 类 并 不 支持 该 模式 。 这 里 只 需要 使 用 preg_replace 水 数 处 理 即 可 。 对 应 的 index.html 视图 
模板 如 以 下 代码 所 示 。 


EE E 


<div class="content_left"> 
<div class="hm1"> 


<span> 查 看 :<!--{Scontent .browse_number }--></span> | <span> 
回复 :<!--{$SreplyCount}--></span> 
</div> 
<div class="user head"> 
<img src="<!—-{$content.add_user |uidToData=ł##, "head_image"} 
——>" width="100" height="100" border="0" /> 
</div> 
<div class="hm2"> 
<span><li><!--{user.topicNumber}--></li> È W </span><span><li> 
<!--{S$Suser.integral}--></1i> 积 分 </ span> 
</div> 
<div class="hm3"> 
<1i> 称 呼 : <!--{Suser.user nickname}--></1i> 


list < condition weontentl' oender eg 14 
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<else /> E 


</if></1i> 
<1i> 积 分 : <!--{S$Suser.integral}--></1i> 
<1i> 帖 子 : <!--{S$Suser.topicAndreplayNumber}--></1i> 
<1i> 注 册 日 期 </1i> 
<li-<1 (fuser. reg timeldate ry md H:i:s",###}--></li> 
<1i> 最 后 登录 </1i> 
<Li><I {oUt last login Cimeldate=" EE = 
SIE 
</div> 
</div> 
<div class="content_right"> 
<div class="_bar"> 
<em class="wei ico"><span> 分 享 到 ;: </span> 
Dan < id Voo anka eo ee e a a a 
<span><a id="sinaLink" title=" $H RIM RIK EE" href="#"></a> 
</span></em> 
<span> 发 表 时 间 : <!--{$content.add_time |date="Y-m-d H:i:s",###}--—> 
</span></div> 
<div class=" content"> 
<!—-{$content .content }--> 
</div> 
</div> 
</div> 
</div> 
</elt> 


<include file="Content:replylist" /> 
</div> 


17.5.2 ”回复 列表 


在 前 面 的 index.html 代码 中 ， 已 经 使 用 的 include 标签 包含 replylist 模板 。 该 模板 即 为 正 
文 下 的 回复 列表 。 人 代码 如 下 所 示 。 


<volist name="reply" id="vo" key="k"> 


<oliv elass="post" Lo=Wreply <!=={5y0., LO) ==>" > 

<p class="meta"><span class="date"><?php echo ($firsRowt$k);?># 楼 </span> 

<epam class "admin L a bref T javascript shon elisi. = iovo id] e> 
> 

<span class "golist" <a href "top mm a /aan> 


<div class="topicblock" > 


<li><a href='_ PROJECT_URL_ ?m=admin&a=reply&rid=<!--{Ş$vo.id}-->&p=<!- 
—-{$_GET.p}-->&cid=<!--{$_GET.cid}-->'> 修 改 </a></1i> 

<li><a href=' PROJECT URL ?m=admin&a=deletereply&rid=<!--{Ş$vo.id}-- 
zt onclick="return ConfirmDel()"> 删 除 </a></1Li> 

</div> 

</p> 


<script type="text/javascript"> 
function showReList (id) { 


CT 国 547 


PHP MVC 开发 实战 


$("#reply_"+id+" .topicblock").toggle(); 
} 
</script> 
<div class="main"> 


<div class="content_left"> 


<div class="user_head"> 


<img src="<!—-{$content.add user|uidToData=###, "head image"} 
——->" width="100" height="100" border="0" /> 
</div> 


Se DEE 


RE 
有 
</div> 


<div class="hm3"> 
ERAN Ce <!——{S$vo.add user |uidToData=###, "user_nickname" }—-></li> 


<1i> 性 别 : <if condition="uidToData ($vo['add_user'], 'gender') 


eg On 
< e 
/I /> 
<1i> 积 分 : <!--{S$Suser.integral}--></1i> 
<li>ħi F: <!--{S$Suser.topicAndreplayNumber}--></1i> 
<]i3 注 册 目 期 </1i3 
<li><?php echo date("Y-m-d H:i:s",uidToData ($vo['add_user!'], 
reg time”) ) CEE) LS 
<1i> 最 后 登录 </1i> 
<li><?php echo date ("Y-m-d 
Weiss“  uicdtopDarca (Syo ace user], last login time) ) p 2></1Li> 
</div> 


</div> 
<div class="content_right"> 
<div class=" barn><span> 回 复 时 间 : <!--{S$vo.add time|date="Y-m-d H:i:s", 


###}--></span></div> 


<div class=" content"> 
<l— {SVvO. reply Concent ==> 
</div> 
</div> 
</div> 


</div> 
</volist> 


上 述 标 粗 代码 表示 volist 循环 数据 来 自 于 reply 变量 ， 该 变量 需要 在 index 动作 中 定义 。 
如 以 下 代码 所 示 。 


/fe 省 略 部 分 
// 回 复 
了 
$count = ŞTrOpLe->gerReplybise EE 
SPade = new Page ($count, Gil: 
$show = SPage->show() ; 
Sreplv List = Ştopic->limit ($Page->firstRow.!','.Ş$Page->listRows)- 


>getReplyList ($Ş$_GET["cid"]); 
sthis->assign ("firsRow",Ş$Page->firstRow); 
if (C ("URL_MODEL") ==4) { 
Şarrl = array ( 
Ve (NO) NREC lw, 
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"/\?\&p=(\d)/i" 
Is 


Sart? = array ( 
UU Ve 
UAA ep dE 
) 7 
Şshow = preg_replace ( S$arrl, Sarr2, S$show ) ; 


} 
Šthis->assign ("page", $show); 
$this->assign ("reply",$reply_list); 
Şthis->assign("replyCount", $count); 
/7 省 略 部 分 
与 小 版 块 分 页 设计 一 样 ， 系 统 文 持 自 定 义 的 UrIRewriter 模式 ， 而 ThinkPHP 内 置 的 分 页 


扩展 类 并 不 文 持 该 模式 ， 这 里 只 需要 使 用 preg_replace 函数 处 理 即 可 。 

人 至此， 帖子 正文 模块 就 开发 完成 了 。 但 需要 注音 的 是 ， 由 于 UEditor 编辑 需 文 持 插 入 腺 
生 代码 ， 要 在 内 容 页 面 中 实现 相应 的 插入 效果 ， 还 需要 引入 官方 提供 的 CSS 文件 及 编写 
JavaScript 处 理 脚本 ， 如 以 下 代码 所 示 。 


< CVS EE 
// 为 了 在 编辑 器 之 外 能 展示 高 亮 代码 
SyntaxHighlighter.highlight (); 
// 调 整 左右 对 齐 
for(var i=0,di;di=SyntaxHighlighter.highlightContainers[i++];){ 
var tds = di.getElementsByTagName ('td');}; 
for(var j=0,li,ri;li=tds[0].childNodes[j];J++){ 
pi = tas]. tirstOhniLla.chiLlelNodes i g] 
pi. style- hnheighar = li. style height = ri ofrsetcheight r “ORY? 


} 


var editor a = new baidu.editor.ui.Editor(); 
editor a.render( 'myEditor' ) ; 

</script> 

CH A = 

最 终 的 效果 如 图 17-23 所 示 。 


Li javascript- BBS 这 区 


E. e tplocalhost/thread-38-1.h 


EAHA: 21301-15 HS5 


script> 


alert{" fds"); 


1 
2 
3 
D 
5 


«i soript>» 


图 17-23 原生 代码 格式 显示 效果 
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17.5.3 回复 帖子 


用 户 在 浏览 正文 时 ， 可 以 在 帖子 帮 部 迅速 找到 回复 帖子 的 入 口 。 只 需要 在 index.htm X 
件 压 部 适当 位 置 包含 回复 编辑 各 负面 即 可 ， 如 以 下 代 人 码 所 示 。 

<include file="Content:postreply" /> 

对 应 的 bbs/Tpl/Content/postreply.html 文件 代码 如 下 所 示 。 

0 省 略 部 分 

<form method="post" action=" __PROJECT_URL_ ?m=posttopic&a=addreply&cid=<!-- 


{$_GET.cid}-->&id=<!-—-{$_GET.id}-->&p=<!--{$_GET.p}—-->" name="reply_from"> 
<taglib name='Bbs' /> 


<bbs:editor id="textContent" editurl=" BBSPUBLIC /Reso 
urces/ueditor/" config="editor_config_2" width="650" height="220" > 
<!—-{$reply_content }--> 


</bbs:editor> 


<div class="addReply"><input type="submit" name="addReply" 
value=" IO Bn /><span style="margin-left: 8px"> EXA ENK ARR HAEE, WE a iE A E R 
</span> 


</div> 
</form> 
//…… 和 省 略 部 分 
如 上 述 标 粗 代码 所 示 ， 这 里 使 用 了 定制 的 UEditor 编辑 器 ， 配 置 文件 为 editor_config ` 
2.js。 用 户 单 击 “ 回 复 ” 按 钮 后 ， 将 提交 到 PosttopicAction 控制 器 下 的 addreply 动作 中 处 理 。 
代码 如 下 所 示 。 


public function addreply  (){ 


if (Şthis->userlogin==1){ 
SreplwcchM (Reply); 

load ("extend"); 

Şconfig=C ("SCENARIO CONFIG"); 


if ((ccStrLen (trim($_POST["textContent"])))<Ş$config["user_addreply_last_word_count"]){ 
ochire asci ga Content i urien code El Ee ER en H 
$this->error (" 回 复 内 容 最 必须 多 于 ".S$config["user_adqreply_ last_worad_countn"] ." 个 
字符 ") ; 
i 
Le (emacs Gmrlvenewv]|,)») 1 


O y 


$Sthis->error (ne Enn ， 

} 

ŞUser=D ("User"); 

$user_data=$User->UserData (session ("user_id")); 

Şdata=array ( 
Ureply Concent t=>9_POST | Utrext Concenmt™ ], 
eene Eege hee (9$_CET eic] Y, 
Ucacegory ls>lneval (9_GBT | S Law); 
"add time"=>time(), 
"add user"=>S$Suser datal"id"], 
lle > client pi, 
"stats"=>$config["user addreply_status"] 
); 

if ($reply->add ($data) ){ 
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S (Vreply ligt YS GETI eic i y NULG) 


Şthis=>assicm ("url PageWRG ($ Girlvenrcvl,Ss sri o] )) g 
$this->success ("回复 成 功 ")，; 
}else{ 


Log: :write ("用 户 在 回复 主题 时 产生 了 一 个 错误 : ".S$replLy->getLastSdq1() ) ; 
$this->error ("数据 插入 失败 ， 请 联系 管理 员 ")， 

} 

}else{ 
$this->error ("请 登录 后 再 操作 "); 

} 


17.6 帖子 管理 模块 


帖子 拥有 者 或 者 系统 指定 的 管理 员 应 该 具备 对 帖子 进行 相应 管理 的 权力 ， 例 如 删除 、 锁 
定 、 加 入 待 审 、 EE a 下 面 将 对 
第 见 的 帖子 管理 操作 进行 


17.6.1 删除 帖子 


通 种 前 合 管 理 员 或 者 版 主 可 以 对 帖子 进行 删除 操作 。 这 里 所 说 的 删除 并 不 特 指 删除 主题 
帖 ， 还 包括 删除 主题 帖 下 的 回复 列表 。 两 种 删除 模式 在 操作 方式 上 其 有 不 同 之 处 ， 下 面 分 列 


1. 删除 主题 

管理 员 单 击 正文 上 的 “管理 ”链接 ， 将 弹出 删除 等 操作 链接 。 对 于 主题 而 言 ， 删 除 主 
题 意味 看 回复 列表 也 失去 了 作用 ， 所 以 也 需要 一 并 删除 。 需 要 注意 的 是 ， 这 里 所 说 的 删除 
是 指 字 面 上 的 ， 事 实 上 为 了 数据 的 安全 ， 通 第 情况 下 都 会 提供 可 逆 操 作 选 项 ， 即 允许 管理 
员 恢 复数 据 (类似 于 回收 站 )， 只 有 后 台 管 理 员 进行 “清空 回收 站 ”， 系统 才 真 正 删除 数据 
库 中 的 数据 。 

帖子 管理 模块 将 使 用 AdminAction 控制 器 实现 所 有 操作 ， 其 中 deletetopic 动作 实现 删除 
主题 操作 ， 代 码 如 下 所 示 。 

S 删除 帖子 

gd 

public function deletetopic(){ 


if (Şthis->userlogin!=1){ 
$Sthnis->error ("请 登录 后 再 操作 ")， 


} 
ŞUser=D ("User"); 
Şuser_data=$User->UserData (session ("user_id")); 
Şconfig=C("TMPL_PARSE_STRING"); 
SEODLESD (UTODLEY) 5 
Şrows=$topic->where (array ("id"=>$_GET["cid"]))->find(); 
if (!Şrows){ 

$this->error (" 数 据 不 存在 ") ; 
} 


| 
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wis delerte tCopichii) 
if (!IAdminPermission(suser datal"id"], "is delete topic"n) ) { 


Sthis->error(n 权 限 出 错 ") ; 


} 


if (empty($_GET["ac"™]))t 
Šthis->display (); 
elseif ($_GET|"ac"] =de) { 


Sreply-M(TRepIy"); 
swhere=array ( 
SEC ict=>inecval (S Gart | Yere] yy 
) 7 
Şcount=$reply->where (Ş$where)->count ();}; 
if ($count) { 
Şrows=$reply->where ($where)->setField("stats",-1); 
} 
if ($topic->where (array ("id"=>intval ($_GET["cid"])))->setField("stats",-1) ){ 
// 删 除 回复 列表 
Ştopic_recycle=M("TopicRecycle"); 
sdata=array ( 
EE Eege 
Trecwvcle Cimeft-zztImetil, 
"operation_user"=>$user_data["id"] 
) 7 
if (!Stopic_ recycle->add(sdata))t 


Log: :write ("用 户 在 删除 主题 时 ， 产 生 了 一 个 错误 : ".S$topic_recycle->get 
Lastsql()); 
} 
Şrow=$topic->where (array ("id"=>intval($_GET["cid"])))->field("category 
Ee eet Ee 
Şthis->assign("url",CategoryURL(Ş$row["category_id"])); 
$this->success ("删除 成 功 ")，; 
}else{ 
Serrgol=ş$tropl Ce- > gerast dol () -4// Y -$reply -gecast sal 1); 


Log: :write ("用 户 在 删除 一 个 主题 是 产生 了 错误 "，, $errsql); 
$this->error ("删除 失败 ， 请 检测 soL 语句 :".S$Serrsql);， 


} 


} 
2. 删除 回复 


删除 单条 回复 比较 简单 ， 只 需要 使 用 ThinkPHP 中 的 连贯 操作 即 可 。 如 以 下 代码 所 示 。 
/** 
* 删除 回复 
a 
public function deletereply(){ 
TE Ee eil 
SUSSer=D (User) 


$user_data=$User->UserData (session ("user_id")); 


Ş$reply=M ("Reply"); 
Şrows=$reply->where (array ("id"=>intval ($_GET["rid"])))->find(); 
if (!S$Srows)f{ 

$this->error (" 数 据 不 存在 ") ; 
} 


GE ee ee ee Ee 
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wig delete reply”) ) 4 
if (!AdminPermission($user_data["id"], "is delete reply"))I 


$Sthis->error ("权限 出 错 ")， 


} 
if (Şreply->where (array ("id"=>intval($_GET["rid"])))->delete()){ 
sthis->success(" 删 除 成 功 ") ， 


}else{ 
ŝthis->error ("参数 错误 ")， 


17.6.2 ”锁定 帖子 


一 些 特 殊 的 帖子 并 不 需要 用 户 参 与 回复 ， 例 如 公告 、 广 告 、 通 知 等 。 这 里 可 以 使 用 锁定 
功能 拒绝 任何 用 户 的 回复 。 锁 定 帖 子 是 可 逆 的 ， 即 管理 员 可 以 在 任何 时 候 进 行 锁 定 ， 也 可 以 
进行 解锁 。 在 后 合 处 理 上 ， 需 要 判断 该 主题 目前 的 锁定 状态 : 如 来 帖子 处 于 锁定 状态 ， 则 执 
行 解 锁 操 作 ， 奋 则 执行 锁定 操作 。 这 就 是 锁定 的 流程 ， 这 里 使 用 locktopic 动作 实现 ， 代 但 
如 下 。 

/** 

* 锁定 帖子 

E 

public function locktopic(){ 


if (Şthis->userlogin!=1){ 
$this->error ("请 登录 后 再 操作 ")，; 
} 
1 EE EE EE 
SEENEN GE 
suser_ data=$User->UserData(session("user idq") ) ; 
SE EE E Ee 
Şrows=$topic->where (array ("id"=>intval($_GET["cid"])))->find(); 
if (!Ş$rows){ 
$this->error (" 数 据 不 存在 ") ; 
} 
e a e EECH A a Eet 
Uisg lock topie) )i 
Le (lAcminaprermiss on (Suger carca Yie], Wis lock coore) ) i 


$Sthis->error ("权限 出 错 ")， 


} 


LE (Srows | Ww exe Wl 
$lock=0; // 解 锁 
}else { 


$lock=1; // 锁 定 

} 

if ($topic->where (array ("id"=>intval ($_GET["cid"])))->setField("lock",$lock) ){ 
Sthis->success (" 操 作成 功 ") ; 

Lelsel 
Log: :write(" 用 户 在 对 帖子 进行 锁定 时 产生 了 一 个 错误 : ". $topic->getLastSqgl()); 
$this->error ("操作 失败 ， 请 联系 管理 员 ")，; 
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} 
}else{ 
$this->error ("删除 错误 ")， 


17.6.3 置顶 帖子 


默认 情况 下 ， 帖 子 的 排序 是 根据 最 新 回复 和 发 表 时 间 进 行 智能 排序 的 。 如 果 需 要 让 帖子 
始终 你 持 在 版 块 的 前 面 ， 只 需要 将 该 帖子 设置 为 营 项 即 可 。 如 以 下 代 人 码 所 示 。 


/** 
* 设置 置顶 
e 
public function settop () { 
if (Şthis->userlogin!=1){ 
$this->error ("请 登录 后 再 操作 "); 


} 
(SE | el IG E 
SsUser=D ("User"); 


Şuser_data=$User->UserData (session("user_id")); 

STODLE=D (YTODLEY) 
Şrows=$topic->where (array ("id"=>intval($_GET["cid"])))->find(); 
if (!$Şrows){ 


Sthis->error(" 数 据 不 存在 ") ; 
} 


EE el rl HE EE el 
EECH 
if ('AdminPermission(Ş$Şuser_data["id"], "is settop topic"))t 


$Sthis->error ("WIRE"); 


} 
LE Ehel A, reelle Ee 


$5top=0; 

}else { 
三 

} 

if (Stopic->where (array ("id"=>intval($_GET["cid"])))—>setField("top", stop) ){ 
Sthis->success ("操作 成 功 ")，; 

}else{ 


Log: :write ("用 户 在 设置 帖子 置顶 时 产生 了 一 个 错误 : " .Stopic->getLastSdql1() ) ; 
$this->error ("操作 失败 ， 请 联系 管理 员 ")，; 
} 


}else { 
$Sthis->error ("错误 错误 ")， 


17.6.4 ”推荐 帖子 
与 设置 置顶 帖子 一 样 ， 推 荐 帖子 只 需要 更 新 recommend 字段 即 可 。 代 但 如 下 。 


/** 
* 设置 推荐 
a 
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public function topicrecommend () { 
we ($ Gar] "cict]){ 
ŞUser=D ("User"); 
Şuser_data=$User->UserData (session ("user_id")); 
STODLEC=D (TOD le 
Şrows=$topic->where (array ("id"=>intval($_GET["cid"])))->find(); 
if (!$rows){ 
$this->error (" 数 据 不 存在 ") ; 
} 


了 
"is_crecommend_topic")){ 
if (!IAdminPermission(suser datal"id"], "is_crecommend_topic")){ 


$Sthis->error ("权限 出 错 ")， 
} 
} 
if ($rows["recommend"])t 
$recommend=0; 
}else { 
Ş$recommend=1; 
} 
if (Ş$topic->where (array ("id"=>intval($_GET["cid"])))->setField("recommend", 
srecommend) ){ 
Sthis->success (" 操 作成 功 ") ; 
}else{ 
Log: :write ("用 户 在 设置 帖子 推荐 时 产生 了 一 个 错误 : ".Stopic->getLastsql()); 
$Sthis->error ("操作 失败 ， 请 联系 管理 员 ")，; 


} 
}else { 

$Sthis->error ("错误 错误 ") 
} 


17.7 WARA 


本 书 11.4 市 已 经 对 行为 进行 了 详细 、 深 入 的 介绍 。 使 用 行为 可 以 方便 地 对 URL 进行 
拦截 ， 然 后 对 所 请 求 的 资源 进行 处 理 。 总 而 言 乙 ， 行 为 就 是 一 种 拦截 机 制 ， 可 以 在 应 用 程 
序 初 始 化 或 任务 结束 等 任何 阶段 触发 拦截 器 ， 接 下 来 将 利用 行为 扩展 ， 实 现 页 面 访问 量 统 
计 功 能 。 


17.7.1 统计 浏览 量 


统计 帖子 浏览 数量 只 需要 更 新 bbs_topic 表 中 的 browse number 字段 即 可 。 这 里 将 使 用 
目 定 义 行为 实现 统计 功能 ， 之 所 以 使 用 行为 实现 ， 是 因为 行为 本 身 是 一 套 灵 活 强大 的 扩展 机 
制 ， 这 对 于 系统 的 后 期 维护 是 非常 有 利 的 。 

首先 在 bbs/Lib/Behavior 目录 创建 一 个 目 定 义 行为 库 ， 并 命名 为 TopicBrowseBehavior。 
代码 如 例 程 17-12 所 示 。 

【 例 程 17-12] bbs/Lib/Behavior/TopicBrowseBehavior.class.php 文件 代码 。 

<?php 
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// 统 计 帖 子 行为 
class TopicBrowseBehavior extends Behaviort{ 
GEOEESEGEESGL sti? 
public function run(&sparams)t 
S$OUERY STRING=$ SERVER ["OUERY STRING"]; 
parse_str ($QUERY_STRING, $arr); 
Ş$this->str=$arr; 
If (strtolower ($this >str|"m"])~= ~ Tcontent!" && Ae >str|"cida"]){ 
$this->statisticsBrowseNumber (); 


} 
/ 大 大 
* 统计 浏览 量 
SE 
private function statisticsBrowseNumber (){ 
Şstr=$this->str; 
ŞTopic=M ("topic"); 


return $Topic->where ('id='.Ş$str["cid"])->setInc('browse_number'); 


} 
} 


Pe 


创建 目 定 义 行为 后 ， 还 需要 在 bbs/Conf/tags.php 配置 文件 中 开局 行为 才 会 生效 ， 人 代码 如 下 。 


'app_init'=>array ('TopicBrowse'), // 帖子 行为 


17.7.2 ”登记 浏览 位 置 


用 户 在 浏览 版 块 时 ， 可 以 利用 目 定 义 行为 统计 用 户 正 在 浏览 的 位 置 ， 方 便 后 台 进 行 数 据 
统计 。 这 里 将 使 用 CategoryBrowseBehavior 行为 实现 ， 代 人 码 如 例 程 17-13 所 示 。 

【 例 程 17-13] bbs/Lib/Behavior/CategoryBrowseBehavior.class.php 文件 代 人 。 

<?php 

/ /浏览 版 块 行为 


class CategoryBrowseBehavior{ 


EE EE 

public function run(&sparams)t 
SOQOUERY STRING=$ SERVER ["OUERY_ STRING" ] ， 
parse_str ($QUERY_STRING, $arr); 
Ş$this->str=$arr; 
if (strtolower ($this->str["m"])=="category"){ 


Şthis->registerThisBrowseCategory ();}; 


| 

private function registerThisBrowseCategory (){ 
E e he 
EE EE Ee 


session ("CategoryUrl",urlencode ($url)); 


} 
} 


行为 扩展 非常 灵活 及 强大 ， 不 仅 可 以 实现 页 面 统计 等 简单 功 
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b. 事实 上 行为 完全 可 以 实 
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现 系统 所 有 基于 URL 拦截 的 功能 ， 例 如 用 户 权 限 控制 、 管 理 员 权限 控制 、 计 划 任 务 等 ， 在 
实际 应 用 开发 中 可 根据 需要 有 灵活 使 用 行为 扩展 。 


17.8 ”帖子 搜索 模块 


论坛 是 一 个 数据 量 比较 集中 的 平台 ， 所 以 提供 站 内 搜索 模块 是 非常 必要 的 。 接 下 来 将 
使 用 两 种 方式 实现 搜索 模块 : 一 种 是 使 用 传统 的 LIKE 得 询 语句 实现 标题 搜索 ; 另 一 种 是 
基于 Coreseek 的 全 文 搜 索 。 无 论 哪 一 种 搜索 模式 ， 系 统 都 文 持 用 户 权 限 控制 。 下 面 分 别 


介绍 。 


17.8.1 标题 搜索 


搜索 模块 使 用 SearchAction 控制 器 实现 。 用 户 可 以 在 论坛 的 任意 地 方 找到 搜索 入 口 ， 并 
且 可 以 选择 搜索 模式 ， 如 图 17-24 所 示 。 


请 输入 关键 字 RETE v GO 


图 17-24 搜索 模式 
选择 标题 搜索 时 将 使 用 title 动作 进行 处 理 。 代 码 如 下 所 示 。 


/ 大 大 

* 标题 搜索 

e? 

public function title(){ 

if (!empty ($_GET["words"])){ 

ŞUser=D ("User"); 
Şuser_data=$User->UserData (session ("user_id")); 
if (!IUserPermission(Suser datal"id"], "is_search_title")){ 


$this->error ("你 没有 搜索 标题 的 权限 "); 
} 
SES Ee 
Şwhere=array ( 
"title"=>array ("like","%".mysql_escape_string($_GET["words"])."%"), 
us tats -ziii 
) 7 
import ("ORG.Util.Page"); 
$count = Ştopic->where (Ş$where)->count (il: 
if ($count) { 


$Page = new Page ($count, 10); 
$show = SPage->show() ; 
$list = Stopic->where (Ş$where)->limit ($SPage—>firstRow.','.S$Page—->1istRows) 


—>select (Oh: 
foreach ($list as $key=>Şvalue){ 
$list I Skeyl [" title" ]=spreg replaces ("/ (19 CBT | words" ] y) ee 
color='red'>\\I<k/font>", ER E e EEN 
} 
SrtnLs=>assiLcMm (Vlist ®; $LIST) 7 
EE EEN 


E En 
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sthis->assign ("navpage", $show); 


}else{ 
$this->error ("没有 标题 天 键 字 为 【".$_GET["words"] ."】 的 记录 ， 可 以 尝试 使 用 全 文 搜索 。" ) ; 
} 
les SS la 
}else{ 
$this->error (" 请 输入 关键 字 ") ; 
} 


} 
对 应 的 title.html 模板 代码 如 下 。 
MAM CHE 
<volist name="list" id="vo"> 
<li class="v_link"> 


<em class="top_head_img"><!--{:TopicHeadImage ($vo['id'],PageURL 
(Svo['id']))}-—></em> 
a ne 2 I or ll Ri 
>" ><!—-{$vo.title}--></a> 
<span> 
<em> 用 户 :<a href=" APP ?m=user&uid=<!--{S$vo.add user}—-— 
| 
date="m H a H H:i", ###}--></em> 
</span> 
SE 
</volist> 
Jp 省 略 部 分 


17.8.2 使 用 全 文 搜索 

使 用 Coreseek 能 够 轻松 地 实现 高 性 能 海量 全 文 搜索 引擎 ， 对 于 论坛 这 种 交互 性 强 的 应 
用 ， 使 用 Coreseek 实现 全 文 搜索 是 最 适合 不 过 了 。 用 户 选择 全 文 搜索 模式 后 ， 系 统 将 引导 到 
fulltext 动作 ， 在 该 动作 中 可 以 完成 全 文 搜索 处 理 ， 如 以 下 代码 所 示 。 


/ 大 大 

* 全 文 搜 索 

E 

public function fulltext (){ 

if (!empty ($_GET["words"])){ 

ŞUser=D ("User"); 
Şuser_data=$User->UserData (session ("user_id")); 
if (!IUserPermission($user datal[l"id"], "is_search_fulltext")){ 


$this->error ("你 没有 全 文 搜 索 的 权限 "); 
} 
Şconfig=C ("SCENARIO CONFIG"); 
load ("extend"); 
import ("Q@.ORG.sphinxapi","",".php"); 
$sp=new SphinxClient () ; 
$conn=$sp->setServer ($config["Sphinx"] ["host"], $config["Sphinx"] ["prot"]); 


$swords=urldecode ( $_GET ["words"] ); 

if (l Srae = $sp->cwery ( Swords, Ysw )) 1 
cie ( YBrrors" 。 $Ssp->getcLasSeolL () ) 7 
exant C ORNE; 


558 O B 


第 17 章 开发 论坛 系统 


importe (TORG. Uti l Bage TA MA 


ŞPage = new Page ($ras["total"],4); 

$ show = $Page->show ();// 分 页 显示 输出 
$sp->setLimits ($Page->firstRow,$Page->listRows); 
SEET 

SSEES = array_keys SEES Timatchesanl ji: 


rk = Mullig 

// 提 示 key 

foreach ( S$strs as $key => $value ) { 
SSE oS WUW a Svali: 

} 

// 得 到 所 有 文章 id 

EE 

/ /但 询 mysql 数据 库 

Se e De ee 

SE Al KA Be UE Re e Ree eeh eh e 

Şrows=$topic->where ($data)->select(); 


/ / RIET zE 
Şopts = array (); 
e V oserore macaa] = “<o><t0nt COlor=V rec t>" s 
Şopts['after_match'] = '</font></b>'; 
foreach (S$rows as $key=>Şvalue){ 
ştitlell = Srowsl|skeyli'ticlet]; 
Seet) i = areg e plaa EE e alur en | se |e 


fenai ENEE EE EE EE Eer lsel naen eae 
EE EE 


["time"] i " 秒 " ， 


} 

$title=$sp->BuildExcerpts ($title, "main", $words, $opts); 
Şcontent=Ş$sp->BuildExcerpts ($content, "main", Swordqs $opts); 
foreach ($rows as S$SK=>SV) { 


EE ISk i ticle] = Sticlelsk]e 
Ş$rows[$k]['contnet'] = $content [$k]; 
} 
/ IRET RAR 
//dump ($rows); 
search info=" 关 键 字 为 【".S$Swords."】 共 有 ".s$rs["total"]." 条 记录 ， 用 时 ".s$rs 


SEE 
Şthis->assign ("navpage", $show); 
StrhLs->assicm DEE Ee ee E 
Şthis->display ();}; 


}else{ 


} 


Stnis->error ("请 输入 关键 字 ")， 


} 
如 上 述 标 粗 代码 所 示 ， 这 里 使 用 sphinxapi.php 扩展 类 操作 Coreseek 服务 器 。 但 是 如 果 


当前 Web 服务 器 已 经 安装 了 Sphinx 扩展 模块 ， 则 可 以 免 去 类 库 导 入 步骤 ， 只 需要 直接 实例 
化 即 可 。 代 但 如 下 。 


$sp=new SphinxClient () ; 


SE Server (Sconi Ee EE ER E Eege GD SEA enee) 


关于 Coreseek 全 文 搜索 服务 器 的 配置 及 使 用 ， 本 书 第 14 章 已 经 有 详细 介绍 ， 这 里 不 再 
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重 述 。 直 接 给 出 配置 文件 数据 ， 如 例 程 17-14 Dro, 
【 例 程 17-14] Coreseek 配置 文件 代码 。 


source main 


{ 


type = mysql 

sql_host = 192.168.010 

sql_user = root 

sql_pass = root_root 

sql_db = bbs_demo 

sql_port = 3306 # optional, default is 3306 
sql_query_pre = SET SESSION query_cache_type=0FF 
sql_query_pre = SET NAMES utf8 


sql_query_pre = REPLACE INTO sph_counter select 1,max(id) from bbs_topic 


sql_query = SELECT id,title,content FROM bbs_topic \ 
where id <= (select max doc id from sph counter 
where counter_id=1) 
sql_attr_uint = add_time 
sql_attr_timestamp = LEl 


sql_query_info SELECT * FROM bbs_topic WHERE id=$id 


index main 


{ 


source = main 
path = /usr/local/coreseek/var/data/main 
docinfo = extern 


charset_dictpath = /usr/local/mmseg3/etc/ 
charset_type = AA CRo OEE 
| 
source mysql topic 1:main 
{ 
sql_query_pre = SET NAMES utf8 
sql_query = SELECT id,title,content FROM bbs_topic \ 
where id >(select max doc id from sph_counter where counter_id=1) 
sql_query_post = REPLACE INTO sph counter select 1,max(id) from tpk_article 


index topic 1:main 
{ 
source = mysqgql topic 1 
path = /usr/local/coreseek/var/data/main mysql topic 1 
charset dictpath = /usr/local/mmseg3/etc/ 
charset type = zh cn.utf-8 


indexer 


{ 


mem_limit = 32M 
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Searchd 
| 
EES = 9312 
ig = /usr/local/coreseek/var/log/ 
searchd. log 
query log = /ust/local/coreseek/var/log/cuery.- Log 
eo = 
WM SEHR 
pid file = /usrt/local/coreseek/var/log/searchel. oic 
maL EES ~ 1000 
seemless rotate = 1 
Breop Tne = = 0 
wne o an 


} 
将 “/usr/local/coreseek/bin/indexer mysql topic 1 -=-rotate” 索 引 命令 保存 到 计划 任务 


表 即 可 实现 实时 索引 。 最 终 论坛 全 文 搜索 效果 如 图 17-25 所 示 。 
Firefox ~ 


"em. BBS 讨论 区 
€) @ tp.localhost/bbs.php?m=search&words @ “r ~ C UR. Z£ <Ctri+P | X ex A D- & mis Bis eis Ais 


A 


论坛 首页 得 程 论坛 教育 论坛 娱乐 论坛 java 搜索 标题 v GO 


ERRIZ, 


上 


关键 字 为 【java】 共 有 23 条 记录 ， 用 时 0.459 秒 


Java 编 程 技术 中 汉字 间 题 的 分 析 及 解决 。 ”用 户 李 开 涌 时 间 :01 月 09 日 17:59 

在 基于 Java 语言 的 编程 中 ， 我 们 经 党 妊 到 汉字 的 处 理 及 显示 的 问题 。 一 大 堆 看 不 情 的 乱码 肯定 
不 是 我 们 原意 看 到 的 显示 效果 ,怎样 才能 够 让 那些 汉字 正确 显示 呢 ? Java 语 言 默认 的 编码 方式 是 
UNICODE, 而 我 们 中 国人 通常 使 用 的 文件 和 数据 库 都 是 基于 GB2312 或 者 BIG5 等 方式 编码 的 ,怎样 : Hi@86055.com 
才能 够 恰当 地 选择 汉字 编码 方式 并 正确 地 处 理 汉字 的 编码 呢 ? 本 文 将 从 汉字 编码 的 常识 入 手 ， 结 合 oC ee 


Java 编 程 实例 ， 分 析 以 上 两 个 问题 并 .… 
最 后 登录 : 2013-01-05 17:19:48 


Java 编 程 技术 中 汉字 问题 的 分 析 及 解雇 ”用 户 李 开 涌 时 间 :01 月 09 日 17:59 退出 登录 
在 基于 Java 语言 的 编程 中 ， 我 们 经 常 语 到 深 字 的 处 理 及 显示 的 问题 。 一 大 堆 看 不 收 的 乱码 肯定 最 新 回复 
不 是 我 们 愿意 看 到 的 显示 效果 ,怎样 才能 够 让 那些 汉字 正确 显示 呢 ? Java 语 言 默 认 的 编码 方式 是 
UNICODE: 而 我 们 中 国人 通常 使 用 的 交 件 和 数据 库 都 是 基于 GB2312 或 者 BIG5 等 方式 编码 的 ,怎样 只 多 许 会 员 资格 浏览 
才能 够 恰当 地 选择 汉字 编码 方式 并 正确 地 处 理 汉 字 的 编码 呢 本 文 将 从 汉字 编码 的 常识 入 手 ， 结 合 BD 
Java 编 程 实例 ， 分 析 以 上 两 个 问题 并 .… 


fwwwwwwww 


bw 
Hl ebe A 


图 17-25 全文 搜索 效果 


全 此 ， 论 坛 系统 网 开发 完成 了 了。 需要 注意 的 是 ， 本 系统 只 是 一 个 示例 ， 并 不 适合 用 在 真 
实 的 线 上 环境 。 谈 者 可 以 继续 完善 ， 使 其 更 加 路 大 、 易 用 。 这 里 提供 几 氮 需要 继续 完善 的 拉 
NEUK: 

> 开 肥 系统 管理 后 合 。 

> 继续 完善 权限 模块 ， 例 如 增加 版 块 管 理 员 、 超 级 版 主 等 控制 功能 。 

> 文件 上 传 模块 需要 继续 完善 ， 特 别 是 权限 控制 方面 ， 避 免 造 成 安全 隐 叫 。 

> 需要 继续 开发 会 员 中 心 的 各 模块 。 
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> 开发 更 加 灵活 及 有 效 的 绥 存 控制 (使 用 S 函数 及 静态 绥 存 实现 )。 

> 整合 更 多 微 博 或 第 三 方 服务 ， 以 实现 多 账号 登录 。 

总 而 言 之 ， 论 坛 系统 是 比较 复杂 及 大 型 的 应 用 ， 需 要 处 理 非常 多 的 细节 (尤其 是 权限 控 
制 方 面 )， 这 些 都 要 求 开 发 者 具备 过 便 的 技术 及 团队 合作 意识 ， 才 能 开发 出 真正 的 社区 交流 
平台 。 实 战 开 发 一 直 是 本 书 的 主旨 ， 相 信 读 者 通过 前 面 的 系统 学 习 ， 对 本 章 内 容 的 理解 并 不 
会 感到 困难 。 


17.9 ”小 结 


本 章 通过 一 个 大 型 的 论坛 系统 平台 ， 详 细 介 绍 了 使 用 PHP MVC 进行 实战 开发 的 全 过 
程 。 本 章 内 容 是 以 本 书 前 面 的 内 容 为 基础 的 ， 例 如 数据 库 CURD 操作 、XML 处 理 、Json 处 
理 、ThinkPHP 扩展 以 及 全 文 索引 等 。 通 过 综合 示例 的 方式 ， 帮 助 读 者 巩固 所 学 知识 。 
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开发 自己 的 MVC 框架 


内 容 提要 

MVC 是 一 种 设计 模式 ， 在 各 种 成 熟 的 面向 对 象 编 程 技 术 中 均 有 相应 的 MVC 框架 ( 平 
台 )。 通 过 本 书 前 面 内容 的 学 习 ， 相 信 读 者 已 经 能 够 深入 领悟 MVC 设计 思想 。 但 是 ， 在 一 
些 特别 的 场合 中 ， 并 非 所 有 MVC 框架 都 适合 ， 这 时 就 需要 手动 定制 自己 的 MVC 框架 ， 以 
实现 团队 之 间 更 好 协作 。 

定制 自己 的 MVC 框架 ， 不 仅 要 求 读者 全 面 理解 MVC 设计 思想 ， 还 要 求 熟悉 并 掌握 各 
种 成 熟 类 库 的 使 用 (例如 Smarty、PHPLIB )， 最 重要 的 是 具备 过 硬 的 PHP 面向 对 象 编程 技 
术 。 所 以 阅读 本 章 ， 需 要 读者 首先 掌握 本 书 前 面 所 有 内 容 。 

本 章 将 通过 开发 一 个 简单 的 MVC 框架 ， 深 入 介绍 MVC 设计 模式 中 的 URL 处 理 、 数 据 
库 处 理 、 扩 展 处 理 等 内 容 ， 帮 助 读者 从 使 用 MVC 的 技术 层面 提升 到 开发 MVC 的 技术 层 
面 。 本 革 内 容 从 理论 到 实践 ， 配 合 简明 的 代码 示范 ， 将 使 读者 全 面 掌握 MVC 框架 的 开发 。 


学 习 目 标 


TERR MVC 框架 的 理论 基础 。 
掌握 MVC 框架 的 文件 结构 及 规范 。 
理解 开发 MVC 框架 中 的 难点 、 要 点 。 
掌握 MVC 框 染 的 文件 载 入 机 制 。 
掌握 MVC 数据 库 操 作 的 实现 过 程 。 
全 面 提高 PHP 的 编程 水 平 。 

© 全 面 提 高 编程 模块 的 整合 技能 。 
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18.1 开发 前 准备 


通过 前 面 内 容 的 学 习 ， 相 信 读 者 已 经 对 MVC 框架 有 了 深刻 的 认识 。MVC 框架 提供 了 
高 效 好 用 的 数据 库 操 作 、 驱 动 扩展 、 类 库 扩 展 等 机 制 ， 为 完成 复杂 的 高 质量 应 用 提供 了 平台 
文 持 。 事 实 上 ， 这 些 看 似 复 杂 的 MVC 功能 模块 设计 ， 在 PHP 开发 中 是 非常 基础 的 ， 一 个 简 
单 的 MVC 框架 甚至 只 需要 编写 少量 的 代码 就 能 实现 。 学 习 本 章 内 容 不仅 可 以 掌握 PHP 
MVC 开发 模式 ， 还 能 设计 、 开 发 、 定 制 
自己 的 MVC 框架 。 


单 和 人口 文件 “上载 和 请求 
18 1 1 开发 MVC ERMER 


开发 一 套 MVC 框架 并 非 想 象 中 的 那 
么 困难 ， 只 要 具备 PHP 面向 对 象 编程 技 
Y， 普 通 的 PHP 程序 员 能 也 完成 简单 的 
框架 设计 。 但 是 ， 很 多 程序 员 甚 至 高 级 的 
PHP 程序 员 ， 和 苦于 找 不 到 设计 入 口 ， 往 往 
最 后 搞 得 力不从心 。 从 根源 上 来 说 ， 这 并 
非 技术 的 问题 ， 而 是 设计 思想 的 问题 。 当 
然 ， 设 计 思 想 是 抽象 的 ， 不 可 能 从 一 开始 
束 能 把 全 部 开发 需要 考虑 在 内 。 这 时 ， 不 
妨 参照 现 有 成 熟 的 PHP 框架 ， 完 成 最 核 
心 的 功能 设计 ， 然 后 在 此 基础 上 衍生 目 己 
的 功能 特性 。 下 面 将 通过 一 张 图 ， 人 简单 理 
解 MVC 框 染 设计 的 基本 思路 ， 如 图 18-1 
所 示 。 

图 中 演示 了 一 个 人 简单 的 MVC 设计 流 
程 ， 当 然 在 实际 应 用 开发 中 ， 也 并 非 一 定 
要 如 此 进行 设计 。 读 者 可 根据 实际 需要 设 请 求 处 理 结 
计 出 符合 要 求 的 设计 图 。 接 下 来 将 通过 真 
实 的 示例 ， 演 示 MVC 设计 的 全 过 程 。 


18.1.2 ”文件 结构 


接 下 来 将 要 介绍 的 MVC Rei, H 
者 为 了 方便 本 书 读者 学 习 ， 临 时 开发 的 一 
个 简单 MVC 框架 ， 并 命名 为 CleverPHP。 
虽然 简单 ， 但 实现 了 MVC 开发 模式 中 比较 重要 的 功能 ， 例 如 URL 处 理 、 模 型 、 视 图 、 控 
制 器 、 连 贯 操作 、 缓 存 、 驱 动 扩展 、 外 部 扩展 等 。 


业务 层 


图 18-1 MVC 设计 流程 图 
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CleverPHP 将 使 用 Smarty+PDO+ 类 工厂 的 形式 开发 ， 此 框架 功能 上 与 主流 的 MVC 框架 
没有 可 比 性 ， 这 里 只 作为 演示 之 用 。CleverPHP 的 文件 及 目录 结构 组 成 如 下 。 


CleverPHP 

Een, tte 存放 基础 类 库 、 函 数 库 
E ee 存放 框架 配置 文件 
上 框架 核心 

HE E 系统 内 置 扩 展 

| E priye wo 驱动 扩展 
EE eer 框架 语言 文件 处 理 
Ee PDo 套件 
D E 第 三 方 扩展 处 理 

| coreseek 

人 视图 引擎 


谈 者 可 以 在 http://beauty-soft.net/book/cleverphp 网 站 页 面 下载 到 CleverPHP 所 有 文件 。 
接 下 来 将 通过 示例 的 方式 ， 介 绍 CleverPHP 所 有 文件 的 作用 与 实现 。 


18.2 ”核心 类 库 


核心 类 库 是 一 个 极其 重要 的 基础 类 库 ， 为 了 方便 介绍 ， 这 里 将 核心 类 库 分 成 多 个 单元 进 
行 介 绍 。 首 先 古 MVC 框 染 的 入 口 ， 即 环境 初始 化 。 


18.2.1 初始化 


主流 的 MVC 框架 都 是 单 入 口 文 件 的 ， 这 样 的 好 处 是 可 以 方便 地 对 整体 环境 进行 控制 、 
定制 等 。 缺 点 是 由 于 一 时 载 入 过 多 类 库 ， 将 造成 一 定 的 性 能 损失 。 但 通常 情况 下 ，MVC H 
架 将 要 求 网 站 开发 人 员 手 动 或 目 动 载 入 所 需要 的 类 ，CleverPHP 采用 的 是 目 动 载 入 机 制 。 这 
些 文件 的 载 入 ， 需 要 在 入 口 文件 中 完成 ， 这 里 将 入 口 文件 命名 为 CleverPHPphp， 人 代码 如 例 
FE 18-1 所 示 。 


【 例 程 18-1] /CleverPHP/CleverPHP.php 文件 代码 。 
<?php 


include_once ("BLL/shortcutinfo.php"); / /导航 快捷 方式 
include 'BLL/function.php'; //A RA 

include 'BLL/constants.php'; 

include ("BLL/app.php"); 

reguire once ("PDO/PDO DB Conn. el non 7/7 Apdo 
?> 


上 述 代 人 码 中 ，CleverPHP.php 入 口 文件 共 载 入 了 5 个 核心 类 库 文件 。 其 中 app.php 是 一 个 
环境 初始 化 文件 ， 负 责 初 始 化 框架 所 需要 的 基本 环境 ， 例 如 载 入 类 库 、URL 处 理 等 。 
前 面 的 操作 只 是 初始 化 CleverPHP 框架 本 身 的 环境 ， 但 程序 员 在 开发 网 站 时 ， 还 需要 对 
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网 站 的 环境 进行 初始 化 。 初 始 化 开发 网 站 将 使 用 run 方法 实现 ， 访 方法 位 于 app AF, RI 
如 下 。 


/ 大 大 

* 初始 化 环境 变量 

大 

SCH 

static public function Run()t 
Şapp=new App () ; 
Şapp->AutoLoadFun (); 
Şapp->NavUrl (); 


18.2.2 URL 处 理 


CleverPHP XHP URL 模式 ， 一 种 是 传统 的 GET 参数 传递 模式 ， 男 一 种 则 是 主流 
MVC 框架 支持 的 PATHINFO 模式 。URL 处 理 在 NavUrl 方法 中 完成 。 代 码 如 下 。 
/** 
* 页 面 导航 模型 
w% Enter description bere, 
SCH 
private function NavUrl(){ 
Lt (lemptv(ë SERVERI PATH INKONI 
Sparray $ SERVER|' PATH NEO): 
Şurl=Ş_SERVER ['REQUEST_URI'!']; 
Şstr=explode ("/",trim($parray,"\/")); 
// 得 到 控制 器 类 名 
EE (empty ($str[0]) ee 
// 得 到 动作 方法 
smodel uetiret (empty (Sstre Index": see 


}else { 
/ /传统 的 GET 传 参 模式 
Scontroller=isset ($_GET["m"])?2ucfirst ($_GET["m"]) ."Controller":"IndexController"; 
smodel=isset ($_GET["a"])?ucfirst (Ş_GET["a"]):"Index"; 


} 
if (@!class_exists($controller)){ 
exit(Scontroller." 控 制 器 不 存在 ") ; 
} 
Ş$this->controller=Ş$controller; 
smodel=empty (Ş$model)?"Index":Ş$model; 
Şthis->model=ucfirst ($model); 
$controllerClass=new $controller(); 
Şur=url_parse(); 
ŞcontrollerClass->$model(); 
} 
最 终 ， 整 个 app.php 文件 代码 如 例 程 18-2 所 示 。 
【 例 程 18-2] CleverPHP/BLL/app.php 文件 代码 。 
<?php 


/7 | 应 用 程序 初始 化 
566 O E 


第 18 章 开发 自己 的 MVC 框架 


// | 加 载 必 要 的 数据 


class App{ 
proeeceed o on roller; 
prorecred ee E 
/** 
* 初始 化 环境 变量 
大 
人 
static public function Run() { 
Şapp=new App (); 
$app->AutoLoadFun (); 
mapp -Navor i); 


i 

/** 

* 页 面 导航 模型 

* Enter description here 

> 

private function NavUrl(){ 

if (!empty($_SERVER['PATH_INFO'])){ 
Şparray=$_ SERVERI PATH _ INFO"]; 
SUurl=$_SERVER['REQUEST URI']; 

$str=explode("/",trim($parray, "\/")); 


SCONcroller=uerirse (emoty (Sscrr lO] ) ? Tance s Ssrri[0]) -VConcroller 


Smede l-uciirstOmpty St index e Scri? 


}else { 
$scontroller=isset ($ GET["m"]) ?ucfirst($ GET["m"])."Controller":"IndexController"; 
smodel=isset ($_ GET["a"]) ?ucfirst($ GET["a"]):"JIndex"; 


} 
if (@!class_exists($controller)){ 
exit (Scontrollez.n" 探 制 器 不 存在 ") ， 
} 
Ş$this->controller=Ş$controller; 
smodel=empty (Ş$model)?"Index":Ş$model; 
$this->model=ucfirst ($model); 
$controllerClass=new $controller(); 
Şur=url_parse(); 
$controllerClass->$model(); 
} 
// BIRA HEX mär 
private function AutoLoadFun () { 
if (file_exists (AppDir."/Common/function.php"))t 
include_once AppDir."/Common/function.php"; 


RE 


18.3 ”控制 颖 的 开发 


控制 占 通 常 只 负责 处 理 URL 请 求 ， 它 本 身 不 执行 重要 的 功能 运 


。 前 面 所 到 过 ， 


CleverPHP 使 用 类 GEHEIT 所 以 需要 在 控制 右 中 完成 文件 的 载 入 。 这 里 将 控制 器 基 类 
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命名 为 Controller， 下 面 对 重 要 的 功能 设计 进行 介绍 。 


18.3.1 类 目 动 载 入 
关 目 动 载 入 对 于 MYVYC RUE, SEI, OEB, 


<?php 


include '/Clever/Bill/Cache.class.php'; 
function index (){ 
Şobj=new Cahce () ; 
// 有 丈 辑 代码 
} 
?> 
上 述 代 人 码 是 传统 PH 开发 中 的 文件 引入 模式 。 但 在 MVC 中 ， 只 需要 直接 实例 化 类 即 
可 ， 而 不 需要 每 次 导入 相应 的 类 文件 。 代 码 如 下 。 


<?php 


function index (){ 
Şobj=new Cahce ();} 
/ EARI 

} 


对 比 两 种 引入 模式 ， 无 疑 后 面 的 代码 更 加 简洁 。 在 MVC 中 ， 要 实现 目 动 载 入 类 文件 ， 
w AWADA: 一 种 使 用 正则 三 找 ;， 另外 一 种 使 用 PHP 提供 的 _autoload 日 动 载 入 函数 。 
目前 主流 MVC 框架 ， 都 是 使 用 _autoload 实现 文件 自动 加 载 的 ， 这 里 只 需要 将 ”autoload Gë 
数 定义 在 公共 函数 库 〈(CleverPHP/BLL/function.php〉 中 即 可 。 代 人 码 如 下 。 


站 = 一 一 一 一 一 一 一 一 一 一 一 一 一 

// | 目 动 加 载 类 文件 

/ /下 一 一 一 一 一 一 一 一 一 一 一 一 一 一 

function _ autoload($className) { 


if (strpos ($className, "Model")>1){ 
include AppDir."/Lib/Model/".ucfirst ($className).".class.php"; 
}elseif (Ş$className=="Model1"){ 
include 'CleverPHP/DALFactory/Model.class.php'; 
}elseif (strpos (ş$className,"Controller")>1){ 
include AppDir."/Lib/Controller/".ucfirst ($className).".class.php"; 
}elseif ($className=="Controller"){ 
include 'CleverPHP/DALFactory/Controller.class.php'; 
}elseif (Ş$className=="Cache"){ 
include ("CleverPHP/DALFactory/Cache.class.php"); 
}elseif ($className=="Log"){ 
include ("CleverPHP/DALFactory/Log.class.php"); 
}elseif ($className=="Session"){ 
include ("CleverPHP/DALFactory/Session.class.php"); 
| 


} 

如 上 述 代码 所 示 ， 当 开发 人 员 试 网 调用 一 个 类 ，PHP 引擎 会 首先 得 找 该 类 文件 是 人 否 已 经 
和 存在， 如 果 不 存 在 将 会 触 友 _autoload 函数 ， 最 后 才 进 行 资源 创建 及 释放 。 也 就 是 说 ， 
_ autoload 是 目 动 触发 的 ， 不 需要 开发 人 员 手 动 调用 。 所 以 ， 只 需要 在 入 口 文 件 中 ，3 引 入 公 
共 类 库 文件 ， 即 可 实现 全 局 类 库 和 日 动 载 入。 
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18.3.2 加载 模板 引擎 (View) 


前 面 提 到 过 ，CleverPHP 的 视图 处 理 引 擎 是 基于 Smarty 的 。 只 需要 在 控制 鼎 中 完成 
Smarty 的 配置 及 初始 化 即 可 。 代 码 如 下 。 


public $smarty; 

/** 

* 初始 化 模板 引擎 

* Qsee IController::View() 
E 


public function View (){ 


include ("CleverPHP/View/Smarty/Smarty.class.php"); 
Şthis->smarty=new Smarty (); 

Şsmarty_config=C("smarty_config"); 
Şthis->smarty->template_dir=Ş$smarty_config[|"template_dir"]; 
Şthis->smarty->compile_dir=$smarty_config["compile_dir"]; 
Şthis->smarty->config_dir=$smarty_config["config_dir"]; 
Şthis->smarty->cache_dir=$smarty_config["cache_dir"]; 
Şthis->smarty->caching=$smarty_config["caching"]; 
Şthis->smarty->left_delimiter=Ş$smarty_config["left_delimiter"]; 
Şthis->smarty->right_delimiter=Ş$smarty_config["right_delimiter"]; 
Schals=>smar ty- > regi ister? lugin (VrwictLlon” W URL Y, WgetcURLY) z 
rals- >gnerty -> register lugin (Y Ee EE EE ; Comi Log” p GY) p 


j 


通过 上 述 代 人 码 ， 在 项 目 控 制 占 中 ， 只 需要 使 用 $this->ViewO 即 可 加 载 Smarty Om OI. 
上 述 代码 使 用 registerPlugin 注册 CleverPHP 函数 库 中 的 函数 ， 这 样 就 可 以 直接 在 视图 模板 中 
进行 调用 了 。 


18.3.3 ”处 理 消息 (message) 


消息 主要 指 操作 成 功 提示 信息 及 失败 提示 信息 。 这 里 统一 使 用 message 方法 处 理 。 代 码 
GANG 


/** 
* 输出 信息 
x Qsee IController::success() 
E 
public function message ($data=array (),$ajax=false, Stype="success") { 
Sthis->View() ， 
if ($ajax==false){ 
/ /标题 
array key exlgts Keen bere E e Eeer Eh El DA Ee EES 
EE Ree Ree ek DE EE eer 
// 信 息 内 容 


array_key_exists("message", 


sdata) ?smessage=$datal[l"message"] :$message=null; 
Şthis->smarty->assign("message", $message); 
// 转 问 
array key pl Se s(n W, 

sdata) ?$url=S$datal[l"url"] :$url=$_ SERVER["HTTIP REFERER"]; 
// 转 问 时 间 
array_key_exists("times", S$data) ?$times=sdatal"times"] :$times=1; 
Şthis->smarty->assign("times",Ştimes); 
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Şthis->smarty->assign("url", $url); 
if ($type=="success"){ 
Şthis->smarty->display ('success.html'); 
}else{ 
Şthis->smarty->display('error.html'); 
} 
}else { 


Ee ane Ee (Sdaca,;, 下 YJsonY) z 


} 
在 项 目 控制 器 中 ， 只 需要 使 用 $this->message (“操作 成 功 ”); 即 可 完成 消 恩 的 处 理 。 此 
外 ， 还 需要 一 个 用 于 处 理 Json 及 XML 的 消 恩 处理 方 法 ， 代 人 码 如 下 。 
/** 
* ajax 返回 
* Enter description bere 
* Qparam unknown_type $data 
x Qparam unknown_type $info 
x Qparam unknown_type $type 
E 
public function ajaxReturn ($data=array (), status, $type="xml"){ 
ii (array key exists (\ricle”,;, Scara) ) 
Gcdataslntutrlenl-z-Gdatalfncirlenl: 
if (array_key_exists ("message", $data)) 
Şdatas ["message"]=$data["message"]; 
sdatas["status"]=$status; 
if (Stype==" json™)t 
// 输 出 json 
header ("Content-Type:text/html; charset=utf-8"); 
exit (json encode (Ş$datas)); 
Jelseif ($type=="xml"){ 
// 输 出 xml 
header ("Content-Type:text/xml; charset=utf-8"); 
exit (xml_encode ($datas)); 


} 
E, Controller EKP aS AARI UFE 18-3 所 示 。 
【 例 程 18-3】 CleverPHP/DALFactory/Controller.class.php 文件 代码 。 


<?php 

乡 !] 十 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 
// | 控制 右 

几 ] 二 一 一 一 一 一 一 一 一 一 一 一 一 一 一 


class Controller{ 


public $smarty; 
/** 
* 初始 化 模板 引擎 
x Qsee IController::View() 
2 
public function View() { 
include ("CleverPHP/View/Smarty/Smarty.class.php"); 
Şthis->smarty=new Smarty () ， 
ssmarty_config=C ("smarty_config"); 
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Şthis->smarty->template_dir=Ş$smarty_config[|"template_dir"]; 
Şthis->smarty->compile_dir=$smarty_config["compile_dir"]; 
SHEET RE EE REDER ENEE Eet Ab 
Şthis->smarty->cache_dir=$smarty_config["cache_dir"]; 
Şthis->smarty->caching=$smarty_config["caching"]; 
Şthis->smarty->left_delimiter=$smarty_config["left_delimiter"]; 
SE EE Cel imicter=$emarcy Conie EE de Elek 
EE 
GEREENT EE EE 


Şthis->smarty->registerPlugin("function"," _URL 


/** 
* 输出 信息 
x Qsee IController::success() 
ER 
public function message ($data=array (), $ajax=false, $type="success")t 
Sthis->View(); 
if ($ajax==false){ 
/ /标题 
array key exists e ee En E ?Sc1it1l1e=ScaralveLcle"] ssticlesnulily 
SE EE RE Eeer (Yricle”,; ek e e 
// 信 息 内 容 


array_key_exists("message", Sdata)?smessage=sdatal"message"|] :$message=null; 
sthis->smarty->assign ("message", $message); 

// 转 问 
array_key exists("url", S$data) ?$url=S$datal[l"url"] :$url=$ SERVERI["HTIP REFERER"]; 
// 转 问 时 间 

array key exiataiiftimesit, Gdatal2ëtimes-SGdatalitimesitl:ztimes-l: 


Şthis->smarty->assign("times",Ştimes); 

Şthis->smarty->assign("url", $url); 

if ($type=="success"){ 
Şthis->smarty->display ('success.html'); 

}else{ 
Şthis->smarty->display('error.html'); 

} 

}else { 
le KE E Ke Ee Ee 1, YJs@mnY) y 


/** 
* ajax 本 | 
* Enter description here 
* Qparam unknown type Sdata 
x Qparam unknown type $info 
x Qparam unknown type $type 
e 
public function ajaxReturn ($data=array (), status, $type="xml"){ 
we (array key exists (Yriclet, Scara) ) 
sdatas["title"|=$datal"title"]; 
if (array_key _ exists ("message", S$data)) 
sdatas["message"]=$datal"message"]; 
sdatas["status"]=$status; 
if ($type=="json"){ 
// 输 出 json 
header ("Content-Type:text/html; charset=utf-8"); 


E 571 


PHP MVC 开发 实战 


exit (json encode (Ş$datas)); 
}elseif ($type=="xml"){ 
/ /输出 xml 
header ("Content-Type:text/xml; charset=utf-8"); 
exit (xml_encode ($datas) )，} 
} 
} 
public function _ call ($method, $args) { 
exit (Smethod." 方 法 不 存在 " ) ; 
} 


} 


?> 


18.4 REFER 


IT, EEEH TEREG ERO, KARE nl E N lé E Cem Di e, 
模型 的 设计 主要 包括 初始 化 数据 库 张 动 ， 数 据 表 映射 以 及 数据 库 的 CURD 操作 。 下 面 分 别 


der 


18.4.1 使 用 PDO 


PDO 套件 是 一 僚 功 能 强大 ， 人 简单 多 用 的 数据 库 操 作 中 间 件 ， 内 置 了 主流 数据 库 张 动 文 
持 ， 从 PHP 5.1 开始 ，PDO 已 经 作为 插件 整合 到 了 PHP ZRET. KF PDO BREH, 
不 是 本 章 介 绍 的 内 容 ， 读 者 可 以 从 http:/www.php.net/manual/en/book.pdo.php 网 页 中 获得 帮 
助 信息 。 这 里 只 介绍 将 PDO 整合 到 MVC 框架 中 的 全 过 程 。 

站 和 完 创 建 一 个 数据 库 连 接 类 ， 为 了 方便 介绍 ， 这 里 只 需要 使 用 PDO 套件 (不 使 用 传统 
的 mysql_content KZO 即 可 ， 所 以 命名 为 PDO_DB_Conn。 代 人 码 如 例 程 18-4 所 示 。 


【 例 程 18-4】 CleverPHP/PDO/PDO_DB_Conn.class.php 文件 代码 。 
<?php 


class PDO_DB_Conn{ 
static public $PDOs; 


/ 大 大 

* 返回 数据 连接 对 像 

K 

x Q@Qreturn PDO Mysql 

a 
function PDO_DB_Conn () { 

tey d 
Selo _Nnost=C ("alo NOST) s 


$db_name=C ("db_name"); 


self: :SPDOs=new PDO ("mysql :host=$db_host ; dobname=$db_name" ,C ("db_user") ,C ("db_pwd")); 
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self: :ŞPDOs->query ("set names utf8"); 


}catch (PDOException $e){ 
print $e->getMessage(); 
} 


l 


如 上 述 标 粗 代码 所 示 ， 表 示 使 用 mysql 驱动 。 如 果 需 要 切换 其 他 数据 库 驱 动 ， 可 以 根据 
PDO 手册 进行 设置 。 完 成 上 述 步 又 后 ， 只 需要 在 Model 模型 其 类 中 实例 化 即 可 。 代 人 码 如 下 。 
public function _ construct ($tableName=""){ 


if (empty ($tableName)) { 
Sthirs=>TableName=C (Yelo cable prerix™) .Scr reDlace ("Modelv, VW, ger calleci 


elass O7 
}else{ 
Şthis->TableName=C ("db_table_prefix").ŞtableName; 
} 
new PDO _ DB_Conn();} 


} 
这 样 ， 当 Model 类 库 被 继承 或 者 实例 化 时 ，Model 类 将 完成 所 有 数据 库 连 接 准 备 工作 。 


18.4.2 ”模型 实例 化 
模型 通常 是 在 控制 器 中 进行 调用 的 。 例 如 调用 上 述 基 类 模型 ， 代 码 如 下 。 


<?php 
class IndexController{ 
function index (){ 
smod=new ("Model"); 
var_dump (Smod) ; 
} 
} 


为 了 方便 操作 ， 可 以 使 用 快捷 函数 实现 快速 实例 化 模型 ， 这 里 创建 的 快捷 函数 为 MOD, 
代码 如 下 。 
/** 


* 快捷 实例 化 模型 

= Enter description bere ooo 

* Qparam unknown type SMode Name 

E 

function MOD (SModelName) { 
$ModelName=ucfirst ($ModelName); 
$ModelName=$ModelName."Model"; 
return new $ModelName (); 


} 
框架 快捷 函数 可 以 使 用 一 个 独立 的 文件 存放 。CleverPHP 使 用 shortcutinfo.php 文件 存放 
所 有 快捷 函数 。 


18.4.3 ”实现 连贯 操作 


习惯 使 用 第 三 方 MVC 框架 的 读者 相信 已 经 对 连贯 操作 非 芝 熟悉。 连贯 操 作 主 要 
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据 库 增 、 删 、 改 、 碍 的 一 种 操作 方式 《并非 只 适用 于 数据 库 操作 ， 同 样 适 用 于 其 他 类 库 )。 
由 于 连贯 操作 在 方式 上 非常 接近 传统 的 SQL， 但 又 提供 了 全 面 问 对 象 编程 的 特性 ， 对 后 续 扩 
展 提 供 了 良好 的 支持 ， 所 以 越 来 越 多 的 PHP MVC 框架 都 内 置 了 连贯 操作 支持 。 

连贯 操作 主要 是 使 用 _call 魔术 方法 实现 的 。_call 方法 是 一 个 实用 的 函数 ， 能 够 实现 
自动 回调 对 象 方法 。 假 设 PHP 代码 如 下 。 


<?php 
Glass cest 1 


function index (){ 

echo indes o T; 

} 
} 
stestOBJ=new test () ; 
echo $testOBJ->home (); 
?> 
上 述 代码 明显 会 出 错 的 ， 因 为 testOBJ 对 象 中 并 没有 home 方法 。 但 使 用 _call 方法 后 ， 

WÉI Je dr home 方法 ， 解 释 引擎 也 不 会 报错 。 如 以 下 代码 所 示 。 

<?php 
class test{ 

function index (){ 

return ‘index DA"; 
} 
public function _ call ($method, $args) { 
return “你 所 调用 的 ' .$method.' 方 法 不 存在 '; 

} 
} 
stestOBJ=new test () ; 
echo $testOBJ->home () ; 
2> 


ERRER EERI RAAR home 方法 不 存在 ” 从 这 里 就 可 以 出 __ call 方法 的 
作用 。 可 以 利用 __call 方法 的 特性 ， 将 不 存在 的 方法 转换 为 SQL 查询 语句 ， 即 可 实现 连贯 操 
AEs On as 


protected $options = array (); 
public function _ call ($method, $args) { 


s(n EE { 
连贯 操作 的 实现 
SE >options[strtolower (Smethodq)] = S$args[0]; 


return $this; 
}else{ 

exit ("当前 模型 不 存在 " . smethodq. "方法 ") ; 
} 


} 

上 述 代 码 的 作用 是 ， 当 开发 人 员 试 图 使 用 连贯 操作 对 Model 基 类 进行 CURD 操作 时 ， 连 
接 操 作 方 法 中 包含 'field'、'where'、'order、'limit 方 法 时 ， 由 于 这 些 方法 是 不 存在 的 虚 方 法 ， 所 
以 使 用 _call 魔术 方法 将 方法 参数 转换 为 SQL 字符 串 ， 并 保存 到 $options 类 成 员 和 变量 中 。 当 调 
用 真正 存在 的 实 方法 时 ， 根 据 变量 中 的 语句 来 进行 相应 的 操作 ， 例 如 更 新 或 否 询 等 。 


18.4.4 ” 读 取 数据 


前 面 介绍 了 连贯 操作 的 实现 原理 。 连 贯 操作 中 可 以 存在 多 个 虚 方 法 ， 但 实 方法 必须 存在 
574 国 国 
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(只 能 有 一 个 )， 所 有 的 CURD 操作 必须 要 在 实 方法 中 完成 。 例 如 需要 读 取 数据 ， 就 必须 要 


创建 一 个 真实 的 读 取 数据 方法 ， 在 该 方法 中 调用 PDO 相应 的 查询 方法 即 可 。 这 里 将 使 用 
read 方法 实现 数据 查询 操作 。 代 人 码 如 下 。 


/** 


* 读 取 数据 


x Qsee IModel: :read() 


SS 


public function read() { 


if 


(Spdo=PDO_DB_ Conn: :SPDOS) { 

if (array_key_exists("where", $this->options)){ 
swhere=" where ".Ş$this->options["where"];}; 

}else{ 


Şwhere=null; 
} 
Le larrcay key exiscsl(Yorcer",;, Srhilgs=>00r Lons) ) 1 
Sorcder=" order Y. rtrhigs=>0prtlons | Vorder]; 
}else{ 
$order=null; 
} 
Te (Bey key exis (Y LLUnleW, SrALg=>0prLongs) ) 
En limiete Y. SEnis Sodions |" Limit” i]s 
}else{ 
Slimit=® Limie 0, 20W95 
} 
Le (larrcay key exiscs(Yriele, $rnis=>0prLons) ) 1 
SEET E Lons Re AC 
Lelsel 
SE 
} 
LE (!Șthnis=>s@l) < 


Şthis->sql="select ".$field." from ".şthis->TableName.Ş$where.Ş$order.Ş$limit; 


} 

Şpdo->query ("set names utf8"); 
Şrs=$pdo->prepare ($this->sql); 
Ş$rs->execute();}; 
Şthis->lastsql=$rs->queryString; 
return S$rs->fetchAll (); 


}else { 


} 


return false; 


read 方法 利用 PDO 套件 提供 的 fetchAll 方法 ， 实 现 数据 批量 得 询 功能 。 为 了 提高 效 


率 ， 这 里 还 将 创建 find 方法 ， 该 方法 上 只 谈 取 单条 数据 。 代 码 如 下 。 


/** 


* 读 取 单条 数据 


x Qsee IModel: :read() 


E 


public function find(){ 
if ($pdo=PDO_DB_Conn::$PDOs) { 


if (array_key_exists("where", $this->options)){ 
swhere=" where ".Ş$this->options["where"];}; 
Lelsel 
Şwhere=null; 


国 Er 
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} 
li (array key exiscs (Vorder, Srnlg=>00riLong) ) 
Sorcer=Y order W, Srtrhis=>0pt Lons] Vorder]; 
Lelsel 
sorder=null; 
} 
S11imlesv limite 0, 1w} 
LE (array key exisecslYiriele, Stnis— >oions))1 
SrilelgsY W ernis- SareeLonmsl ietsSleW) ew Ws 
Lelsel 
Ee 
} 
Lii (!$this=>sal) 1 
Şthis->sql="select ".s$field." from ".şthis->TableName.Şwhere.Ş$order.Ş$limit; 
} 
Şpdo->query ("set names utf8"); 
$rs=$pdo->prepare ($this->sql); 
Ş$rs->execute();}; 
$rs->setFetchMode (PDO::FETCH_ASSOC); 
Şthis->lastsql=$rs->queryString; 
return Şrs->fetch(); 
}else { 
return false; 


} 


} 
为 了 方便 数据 分 页 ， 还 需要 创建 一 个 统计 方法 ， 并 命名 为 count， 代 人 码 如 下 。 
/** 
AA 
nz Breturn number boolean 
ES 
public function count () { 
if (Şpdo=PDO_DB_Conn::$PDOs) { 
if (array_key_exists("where", $this->options)){ 
swhere=" where ".Ş$this->options["where"]; 
Lelsel 
Ş$where=null; 
} 
LE EE Ee Eed 
Şthis->sql="select * from ".şthis->TableName.Ş$where; 
} 
Şpdo->query ("set names utf8"); 
Şrs=Ş$pdo->prepare ($this->sql); 
Ş$rs->execute();}; 
Scount=$rs->rowCount ();}; 
Şthis->lastsql=$rs->queryString; 
return $count; 
}else { 
recura reales 


18.4.5 ”插入 数据 
在 PDO 操作 中 ， 插 入 或 更 新 数据 可 以 使 用 execute 方法 ， 接 下 来 将 使 用 execute 方法 实 
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现 Model 中 的 数据 插入 功能 。 这 里 将 使 用 create 方法 作为 Model 基础 模型 数据 插入 操作 的 方 
ZZ, En F a 


/** 
* 插入 数据 
* @see IModel::create() 
SC 
public function create () { 
Şdata=$this->options["data"]; 
$field=null; 
Ş$val=null; 
Li (Gdatail 
foreach ($data as Skey=>svalue) { 
if (is_string ($value) ){ 
Şvalue=mysql_escape_string ($value); 
Dame MMS va MEn; 
}else { 
Şvalue=intval ($value); 
} 
Şval =Şyalue, "vs 
Stie OKey ae; 
} 
SS El KE E e Bel Ree 


SE EK EE REN Ee Y, w) s 
SH NEE E EE Ree Ee le e EE 
}else{ 
exit (" 数 据 为 空 ") ; 
} 
if ($pdo=PDO_DB_Conn::$PDOs) { 
Şrs=$pdo->prepare ($this->sql); 
return $rs->execute ();}; 
}else { 
Şthis->error=$pdo->errorInfo(); 
return false; 


} 
} 
U EIRIK, Zehat A E A rent Tt LC ol. MEAE THE a EENE 
中 的 数组 键 值 对 转换 成 SQL 操作 中 的 字段 与 字段 值 。 转 换 的 方式 有 很 多 种 ， 这 里 只 使 用 人 简 
DI foreach 语句 过 历 实现 。 


18.4.6 ”更 新 数据 


更 新 数据 与 插入 数据 在 形式 上 是 一 致 的 。 不 同 之 处 在 于 SQL 语句 的 转换 。 这 里 将 使 用 
update 方法 实现 数据 更 新 ， 代 码 如 下 。 


/** 

* 更 新 的 数据 

* Qsee IModel::update () 

a 

public function update () { 
swhere=" where ".Ş$this->options["where"]; 
Şdata=$this->options["data"]; 
$sfield=null; 
EE EES E 


E Ere 
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foreach ($data as Skey=>svalue) { 
if (is_string ($value) ){ 
Şvalue="'Şvalue'"; 
}else { 
svalue=intval ($value); 
} 
Srielelh =Y; vv Skey. Y =Y .Svan 
} 
EE 


ssets="set ".Ş$sets; 
Şthis->sql="Update ".Şthis->TableName." S$sets"." 5Swhere "; 
}else{ 


exit (" 数 据 为 空 ") ; 
} 
if ($pdo=PDO_DB_Conn::$PDOs) { 
Şrs=$pdo->prepare ($this->sql); 
return $rs->execute(); 
}else { 
Şthis->error=$pdo->errorInfo(); 


return false; 


18.4.7 ”删除 数据 
删除 数据 使 用 delete 操作 方法 实现 ， 代 码 如 下 。 


/** 
* 删除 数据 
= macer OSCILLAON NSFS ooo 
e 
function delete(){ 
if ($pdo=PDO_DB_Conn::$PDOs) { 
if (array_key_exists ("where", $this->options)){ 


swhere=" WHERE ".Ş$this->options["where"]; 


Şthis->sql="DELETE FROM {Ş$Şthis->TableName} ".Şwhere; 

}else{ 

Sthnis->error="where FA RENT"; 

return false; 
} 
Şrs=Ş$pdo->prepare ($this->sql); 
Şthis->lastsql=Ş$Şrs->queryString; 

return $rs->execute () ; 
}else { 


percura EE 


} 
这 里 只 是 介绍 最 基础 的 数据 库 操 作 ， 读 者 还 可 根据 实际 需要 ， 设 计 更 加 好 用 的 CURD 
操作 。 最 终 ，Model 基础 模型 代码 如 例 程 18-4 所 示 。 

【 例 程 18-4] CleverPHP/DALFactory/Model.class.php 文件 代码 。 

<?php 
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// | 实现 Model 


class Model { 
See 
public $TableName; 
public SS1aSESG 5? 
public $error; 


protected $options = array (); 
public function _ call ($method, $args) { 
if (in_array (strtolower (Ş$method),array('field', 'where', 'order','limit'),true)) { 
// 连 员 操作 的 实现 
sthis->options[strtolower(s$method)] = Sargs[0]; 


return $this; 


}else{ 
exit ("当前 模型 不 存在 " . smethodq. "方法 ") ; 


} 


public function _ construct ($tableName=""){ 
if (empty ($tableName)) { 
Sthris=>TableNñane=C (Yelo cable prerii“). ste replace (Mooelv, WWW, ger calleci 
elass 全 到 天 > 
}else{ 


Şthis->TableName=C ("db_table_prefix").ŞtableName; 


} 
new PDO _ DB_Conn();} 
} 
/** 
* 插入 数据 
* @see IModel::create() 
2 
public function create () { 
Şdata=$this->options["data"]; 
sfield=null; 
$sval=null; 
Li (SOaca){ 
foreach ($data as Skey=>svalue) { 
if (is_string ($value) ){ 
Şvalue=mysql_escape_string ($value); 
Svalune=" ! BE 
}else { 
Şvalue=intval ($value); 
} 
Şval =Şyalue, Tu "ş 
Ee 
} 
Se nme r Lelo Y, We 
ŞSvalsrcrriim(Sval, Yw) g 
Sale EE eebe EE (Şrield) Y.” Values (Sval) vp 
}else{ 
exit ("HJN T") 
} 
if (Şpdo=PDO_DB_Conn::$PDOs) { 
Şrs=$pdo->prepare ($this->sql); 
return $rs->execute ();}; 


}else { 
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$this->error=$pdo->errorInfo(); 


recura EaLSEeS 


} 
/** 
* 更 新 的 数据 
* Qsee IModel::update () 
SE 
public function update () { 
swhere=" where ".Ş$this->options["where"]; 
Şdata=$this->options["data"]; 
Ş$field=null; 
LE ($dewra) 4 
foreach ($data as Skey=>svalue) { 
(dn o value) 
Şvalue="'Şvalue'"; 
}else { 
Şvalue=intval ($value); 
} 
Srieleh =Y WW eg 
} 
EE EE Y; YYA 
Sgercs="ser Y, Ee 


SE SE eeh e tchis=>TableName, Soera- w 


}else{ 
SE sen? 

} 

if ($pdo=PDO_DB_Conn::$PDOs) { 
Şrs=$pdo->prepare ($this->sql); 

return $rs->execute () ; 

}else { 

Şthis->error=$pdo->errorInfo(); 


recima rCalsges 


} 
/ 大 大 
* 读 取 数据 
* Qsee IModel::read() 
SH 
public function read(){ 
if (S$Spdo=PDO_DB_ Conn::S$PDOs) { 
if (array_key_ exists("where", $this->options)){ 
swhere=" where ".Ş$this->options["where"]; 
}else{ 
Şwhere=null; 
} 
LE (array key exists Vorcer", Srhis=>0prLong) ) 
SiaeE order W Srmis=->00rL Lons | V orcdert i; 
}else{ 
Ssorder=null; 
} 
EE Seonlsg— Sodelons))l 
S11lmesw limite WV,SEnls— EEN) 
Lelsel 
GE 0, 20W} 


580 O O 
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} 
Tl (array key EE 
Şfield=$this->options["field"]; 
}else{ 
SE 
} 
LE (l$thnis=>s6L){ 
E ".$field." from ".şthis->TableName.Şwhere.Ş$order.Ş$limit; 
} 
Şpdo->query ("set names utf8"); 
Şrs=$pdo->prepare ($this->sql); 
$rs->execute();}; 
Şthis->lastsql=$rs->queryString; 
return $rs-—->fetchAll(); 
}else { 
return false; 


} 
/ 大 大 
* 读 取 单条 数据 
* Qsee IModel::read() 
SS 
public function find(){ 
if (Şpdo=PDO_DB_Conn::$PDOs) { 
if (array_key_exists("where", $this->options)){ 
swhere=" where ".Ş$this->options["where"]; 
}else{ 
Şwhere=null; 
} 
Le (larray key exiscsl EECH 
SOrcder=Y order ns oon nelen 
Lelsel 
Sorder=null; 
} 
Şlimir=" limiete Q, 1 
Le larrlay key exiscs(Yrielel U, Strhilg=>0priong)) 1 
SfEieles W $this=>o0options EE 
Lelsel 
SE 
} 
LE (!$rthnis=>sal) 
Şthis->sql="select ".$field." from ".şthis->TableName.Ş$where.Ş$order.Ş$limit; 
} 
Şpdo->query ("set names utf8"); 
Şrs=$pdo->prepare ($this->sql); 
Şrs->execute(); 
$rs->setFetchMode (PDO::FETCH_ASSOC); 
Şthis->lastsql=$rs->queryString; 
return Şrs-—->fetch(); 
}else { 
return false; 


} 


/** 


Se Ve 
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* @return number |boolean 
E 
public function count () { 
if (Şpdo=PDO_DB_Conn::$PDOs) { 
if (array_key_exists("where", $this->options)){ 
swhere=" where ".Ş$this->options["where"]; 
}else{ 
Şwhere=null; 
} 
LE (l$rtais=>s6lL)1 
Şthis->sql="select * from ".şthis->TableName.Ş$where; 
} 
Şpdo->query ("set names utf8"); 
Şrs=$pdo->prepare ($this->sql); 
Şrs->execute();}; 
Scount=$rs->rowCount (iz 
Şthis->lastsql=$rs->queryString; 
return $count; 
}else { 


return false; 


} 
/大 大 
* 删除 数据 
* Enter description here 
E 
function qelete() { 
if (Şpdo=PDO_DB_Conn::$PDOs) { 
if (array_key exists ("where", $this->options)){ 
swhere=" WHERE ".Ş$this->options["where"]; 
Şthis->sql="DELETE FROM {Sthis->TableName} ".Ş$where; 
}else{ 
$Sthis->error="where FA RENT"; 
return false; 
} 
Şrs=Ş$pdo->prepare ($this->sql); 
sthis->lastSql=$rs->queryString; 
return $rs->execute () ; 
}else { 
return false; 


} 


p> 


18.5 ”扩展 类 库 


扩展 大 致 上 可 分 为 两 关 : 一 类 是 指 在 项 目 中 进行 调用 的 第 三 方 类 库 ， 为 一 类 是 指 框架 本 
吴 所 需要 使 用 到 的 扩展 。 前 者 只 需要 使 用 功能 函数 实现 文件 导入 即 可 ， 但 后 者 需要 编写 相对 
复杂 的 中 间 件 ， 所 以 这 里 主要 介绍 后 者 《 即 张 动 扩展 )。 
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18.5.1 Session 驱动 扩展 


PHP 本 号 提供 了 完善 的 Session 会 话 文 持 ， 由 于 PHP 的 Session 是 基于 本 地 文件 系统 实 
现 的 ， 所 以 并 不 适合 分 布 式 开 发 及 高 性 能 的 运行 环境 。 通 过 Session 扩展 ， 可 以 实现 将 
Session 存放 到 绥 存 系统 或 数据 库 系 统 ， 实 现 高 效 的 Session SE AMARE, ENR 
Session 操作 类 时 ， 必 须要 考虑 可 扩展 的 问题 ， 主 流 的 MVC 框架 通常 使 用 中 间 的 方式 实现 驱 
动 扩 展 。 如 图 18-2 所 示 。 

MVC 内 置 扩展 


调用 中 间 件 处 理 匹配 驱动 类 型 


载 入 所 选择 的 驱动 


选择 一 个 驱动 


A 
询 
数 
据 


中 间 件 处 理 结果 


图 18-2 MVC 扩展 流程 图 


如 图 18-2 所 示 ， 对 于 外 部 程序 而 言 ， 只 需要 实现 公开 的 接口 即 可 。 接 口内 置 了 第 用 的 
操作 方法 ， 无 论 程序 调用 哪个 驱动 ， 实 现 过 程 都 是 一 样 的 。 这 就 是 中 间 件 实现 的 思路 。 

在 Session 中 间 件 设计 中 ， 只 需要 在 外 部 接口 中 实现 普通 的 Session 文 持 ， 以 便 在 基本 的 
PHP 环境 下 也 能 够 使 用 Session。 代 人 码 如 例 程 18-5 所 示 。 

【 例 程 18-5] CleverPHP/DALFactory/Session.class.php 文件 代码 。 


<?php 
//Session 中 间 件 


class Session { 


protected $session_prefix; 


public function Construc ON 


Şconfig=C(); 
if (array_key_exists ("session", Sconfig))t 
Şthis->session_drive=$config["session"]; 
sclassName="Session".ucfirst ($config["session"]); 
$fiel="'./CleverPHP/DALFactory/Extend/Drive/'.s$className.".class.php"; 
LE (ig rile($SrLel)) 4 
GUL once EE el 
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ŞclassName::init();}; 
}else{ 
throw new Exception("{$fiel}"." 文 件 不 存在 ")， 


}else{ 
session_start ();}; 
} 
Şthis->session_prefix=$config["session_prefix"]; 
| 
/** 
* 获取 session 
w Enter description NEPS ooo 
* Qparam string $key 
E 
public function get ($key) 
{ 
Şkey=$this->session_prefix.$key; 
return $_SESSION[Ş$key]; 
} 
/** 
* 5A session 
” Enter description bere, 
= Garam string Il int sname 
* Qparam string $value 
E 
public function set ($key, Svalue)t 
Şkey=$this->session_prefix.$key; 
$_SESSION[$key]=$value; 
if (S$_ SESSION[Skey])t 
ET ile 
}else{ 


recur 07 


} 
/** 
* 清除 session 
=% Enter description NGS ooo 
s Bparam string lorg key 
Ge? 
public function clear (Skey=NULL)I 
Şkey=$this->session_prefix.$key; 
if ($key==null){ 
session_destroy () ; 
yelse { 
waser ($_SBSSIONI Y Skey" i); 


} 
} 


?> 


其 他 的 Session 扩展 则 保存 到 CleverPHP/DALFactory/Extend/Drive 目录 中 ， 并 以 Session 
XXX.class.php 的 文件 命名 规范 作为 Session 驱动 扩展 类 。 例 如 SessionMemcache.class.php 表 
示 使 用 Memcache 作为 Session 驱动 扩展 。 代 人 码 如 例 程 18-6 bro, 

【 例 程 18-6] CleverPHP/DALFactory/Extend/Drive/SessionMemcache.class.php 文件 代码。 
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<?php 
class SessionMemcache { 
private static Shandler=null; 
private static S$lifetime=null; 
private stacie cime = malls 
public static function init (){ 
if(!array_key_exists("session_lifetime", C())){ 
self::$lifetime=ini get('session.gc maxlifetime'); 
| 
sels s $t imest ime (|) 7 
self::$handler= new Memcache () ; 
Şpro=C ("memcache_prot")?C("memcache_prot"):11211; 
self: :Ş$handler->connect (C ("memcache host"), intval($pro)); 
if (!self::Ş$handler) { 
throw new Exception ("Session 驱动 连接 失败 ") ，; 
} 
sellfe Stant Eiter 
} 
public static function start () { 
session_set_save_handler ( 
array CLASS 由 OO 全 下) 
EE 
array (CLASS 
array EE 


KE 


GEET EE 7 
SEET) 


(E 


session_start ();}; 


EE 

* Enter description bere 

* Qparam unknown_type $path 

x @param unknown_type $name 

ER 

public static function open ($path, Şname){ 


Eeturpn rue: 


/** 
* 关闭 
* Enter description here 
E 
public static function close() { 


Eeturpn rue: 


/** 

* 读 取 

E description here 

* Q@param unknown_type S$PHPSESSID 

SR 

public static function read($PHPSESSID)I{ 

Şout=self::Ş$handler->get (self::session_key(ŞPHPSESSID)); 
| e Eaa) 


Perura au 
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return $out; 


/** 

大 EN 

w Enter description NEE ooo 

* Qparam unknown type SPHPSESSID 
x @param unknown type S$data 


SE 
public static function write (S$SPHPSESSID, Sdata)f{ 
Smerchoc=$caca 7? Yser” 3 replace, 


return self: :$handler->Ş$method(self::session_key (SPHPSESSID)， S$data, MEMCACHE 
COMPRESSED, self::$lifetime); 


} 
/** 
EI 
w Enter description bere, 
x Qparam unknown type S$PHPSESSID 
e 
public static function destroy (SPHPSESSID) { 
return self::$handler->delete (self::Şsession_key(ŞPHPSESSID)); 
} 
/** 
* 清空 
=% Enter description bere, 
* Qparam unknown type $lifetime 
E 
public static function gc($lifetime){ 
recur Crue; 
} 
private static function session key ($PHPSESSID){ 
return SPHPSESSID; 
i 
} 
?> 


开发 Session 驱动 扩展 ， 需 要 就 悉 Session get save handler RAJEH. Sol lé: 
基础 上 继续 实现 DbSession.class.hp、RedisSession.class.php 文件 驱动 。 


18.5.2 ”缓存 驱动 扩展 


由 于 Smarty 模板 引擎 本 和 映 就 内 置 了 非常 灵活 及 强大 的 文件 缓存 ， 所 以 这 里 不 再 介绍 文 
件 缓存 的 使 用 。 接 下 来 将 要 介绍 的 是 Memcache 绥 存 扩展 。 当 然 ， 由 于 绥 存 驱动 也 是 基于 中 
则 件 实现 的 ， 所 以 也 可 以 在 此 基础 上 实现 其 他 绥 存 驱动 。 

绥 存 驱动 中 间 件 使 用 Cache 类 实现 ， 在 该 类 中 只 需要 实现 gets sets del 方法 即 可 。 这 
样 ， 网 站 开发 人 员 只 需要 使 用 上 述 3 种 方法 即 可 对 所 有 缓存 张 动 进行 缓存 操作 。 人 代码 如 例 
fE 18-7 所 示 。 


【 例 程 18-7] CleverPHP/DALFactory/Cache.class.php 文件 代码 。 
<?php 


E lm 


class Cache{ 
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protected $cacheName; 
protected $cacheObj; 
EE EE EE EE 
Şconfig=C(); 
LE Sempe EE 
Le (larray key exiscslYcacheY, SCOmMELG)) I 
Şthis->cacheName="Cache".ucfirst ($config["cache"]); 
Lelsel 
Şthis->cacheName="CacheMemcache"; 
} 
}else { 
Şthis->cacheName="Cache".ucfirst ($Drive); 
} 
Şfiel='./CleverPHP/DALFactory/Extend/Drive/'.Ş$this->cacheName.".class.php"; 
LE lig Lle Ek ET 
eovlLre once VI E EA 
$class=$this->cacheName; 
sthis->cacheObj=new $class (); 
}else{ 
throw new Exception ("{$fiel}"." X PEE"); 


} 

/** 
* 设置 缓存 
ara Ke 
SZ Bparam String Sdata 


* Qparam int Sexpire 


we 
public unceltony ser ( key datda o expire 
return $this->cache0bj->setCache ($key, $data, $expire); 
} 
/** 


* 获取 缓存 
* Qparam string Skey 
a 
public function get ($key){ 
return $this->cache0bj->getCache ($key); 
} 
到 大 大 
* 删除 缓存 
* Qparam string $key 
E 
public function aqe1l1(S$Kkey) { 
return $this->cache0bj->delCache ($key); 
} 
} 


?> 


缓存 驱动 类 文件 存放 到 Clever PHP/DALFactory/Extend/Drive 目录 ， 并 以 CacheXXX.class. 
php 的 文件 命名 方式 进行 命名 。 例 如 CacheMemcache.class.php 文件 表示 使 用 Memcache 作为 
绥 存 系统 。 代 码 如 例 程 18-8 所 示 。 

【 例 程 18-8] CleverPHP/DALFactory/Extend/Drive/CacheMemcache.class.php 文件 代码 。 


<?php 
class CacheMemcache{ 
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/大 大 

* Memcache 生成 缓存 

* @see ICache: :Memcaches () 

sy 

protected $memcache; 

runet ion _ Construct (h) 1 
if (Şthis->clientMemcache ()==false) 

die ("Memcache 连接 失败 ")， 

| 

protected function clientMemcache () { 
Şthis->memcache= new Memcache; 
return Şthis->memcache->connect (C ("memcache host"), (int)C("memcache_prot"))? 

true:false; 


} 


public function setCache ($key, $data, Ş$expire=""){ 
EE 
empty (ŞṢŞexpire)?Ş$expire=C("memcache_expire") :$expire=$expire; 


return $this->memcache->set ($key, $data, MEMCACHE COMPRESSED, Ş$expire); 
}catch (Exception $e){ 
exit ($e->getMessage()); 


} 
/ 大 大 
* 获取 缓存 
* Qsee ICache: :getMemcache () 
yA 
public function getCache (Skey)t 
Er 
return S$this->memcache->get ($key,2);，; 
}catch (Exception $e){ 
die ($e->getMessage()); 


} 
/** 
* 删除 memcache 缓存 
* Qsee ICache: :delMemcache () 
SS 
public function delCache ($key) { 
Cry 
return $this->memcache->delete ($key,0); 
bearca (Exception SE) 
die ($e->getMessage()); 


} 
} 


D> 


开发 缓存 驱动 比较 容易 ， 最 简单 的 缓存 驱动 只 需要 实现 setCache、getCache、delCache 
三 种 公开 方法 即 可 。 需 要 注意 的 是 ， 无 论 是 Session 或 者 Cache 驱动 ， 都 必须 配合 项 目 配置 
文件 进行 使 用 。 接 下 来 将 通过 示例 的 方式 ， 全 面 演示 CleverPHP 的 使 用 。 


18.6 测试 MVC 框架 


通过 前 面 的 步骤 ， 一 个 简单 的 MVC 框架 就 完成 了 。 下 面 将 通过 创建 一 个 项 目的 示例 ， 
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18.6.1 创建 项 目 


第 18 章 开发 自己 的 MVC 框架 


CleverPHP 并 不 会 自动 创建 项 目 所 需要 的 目录 及 文件 ， 所 以 在 部 署 时 需要 开发 人 员 自 行 


创建 所 需要 的 目录 结构 。 层 次 如 图 18-3 所 示 。 


AppName 
de Sn 存放 缓存 文件 
E veseeserskesre: 项 目 函 数 库 
Conf ee 存放 项 目 配置 文件 
Lang assssseesssssen 存放 项 目 语言 文件 
= 项 目 核 心 文件 
OD eo te sasse 控制 器 
| :Yodel sssi 目 定 义 模 型 
| Liew weve 视图 文件 
| -一 index 
-一 Runtime seen 编译 文件 


图 18-3 ”目录 结构 


创建 完 相 应 的 目录 结构 之 后 


<?php 
define ("AppDir", "app"); 


， 只 需要 在 网 站 目录 下 创建 入 口 文件 ， 然 后 引入 CleverPHP 
初始 化 文件 即 可 。 这 里 创建 的 入 口 文件 为 index.php， 项 目 命名 为 app。 代 码 如 下 。 


require ("./CleverPHP/CleverPHP.php"); 


App: :Run ();}; 
?> 


为 了 方便 读 示 数据 库 操 作 ， 还 需要 创建 数据 表 。SQL U F o 


-- 数据 库 : `cleverphp` 


— RHA cl content 


CREATE TABLE IF NOT EXISTS 


`cl_content` 


Sigl 


int (11) NOT NULL AUTO_INCREMENT, 


‘title varchar (100) NOT NULL, 


"user 


`content` varchar (200) NOT NULL, 
PRIMARY KEY (`id`), 


KEY 


1@ ( Ler) 


( 


char(50) CHARACTER SET utf8 COLLATE utf8 estonian ci NOT NULL, 


) ENGINE=MyISAM DEFAULT CHARSET=utf8 AUTO INCREMENT=821 ; 
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-- 表 的 结构 "cl user" 


CREATE TABLE IF NOT EXISTS "el user ( 
`id`ò int (11) NOT NULL AUTO_INCREMENT, 
`username` char (20) NOT NULL, 
`password` char (32) NOT NULL, 
"aCe cine Law (ll) NOT NUL, 
PRIMARY KEY (`id`), 
KEY Username ( username ) 
) ENGINE=MyISAM DEFAULT CHARSET=utf8 AUTO INCREMENT=1 ; 
读者 可 以 添加 相应 的 数据 ， 方 便 后 续 的 操作 。 完 成 之 后 ， 在 app/Conf 创建 配置 文件 ， 
并 命名 为 config.php。 代 码 如 下 。 
<?php 
/** 
* 配置 文件 
SE 
Şconfig=array ( 
"db_host"=>"]ocalhost", 
"db_user"=>"root", 
"db_pwd"=>"root", 


"db_name"=>"cleverphp", 


wolo cuert EE EE EE 

ée ee e RE El EA 

"path_mode"=>1, 

"sys_config"=>array ( 

"sys_name"=>"CleverPHP 框架 演示 "， 
ele 

) y 

"memcache_host"=>"127.0.0.1", 

"memcache prot"=>11211, 

"memcache_ expire"=>0, 
"cache"=>"memcache", 
"session"=>"memcache", 

"ESBSLON GDrefixzhozifceiba N; 
"session_lifetime"=>3600, 
); 


R 


至 此 ，CleverPHP 就 准备 就 络 了 。 相 应 的 操作 将 会 在 接 下 来 的 内 容 中 分 别 介绍 。 


18.6.2 Mm CURD 


CURD 操作 是 数据 库 创 建 数据 〈Create)、 更 新 数据 (Update)、 读 取 数 据 (Read), We 
数据 (Delete〉 的 人 简写。 也 是 一 个 框架 最 第 用 的 功能 之 一 ，CURD 操作 的 设计 和 直接 关系 到 网 
站 开发 效率 。 下 面 将 分 别 演示 CleverPHP 的 CURD 操作 。 

1. 增加 数据 

接 下 来 的 所 有 操作 将 在 IndexController 控制 占 中 完成 (所 有 控制 占 需 要 继承 于 
Controller 基础 类 )。 首 先 创建 一 个 Create 动作 ， 在 该 动作 中 只 需要 输出 一 个 视图 模板 即 可 。 
在 该 模板 中 接受 用 户 输入 用 户 名 及 完 码 。Create 动作 代码 如 下 。 


publie fumetion Create () i 
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Sthis->View(); 
Şthis->smarty->display ("create.html"); 
} 


完成 后 之 后 ， 需 要 在 /app/Lib/View 目录 中 创建 相应 的 视 岁 模板 。 这 里 需要 注意 的 是 ， 模 
板 路 径 需 要 巡 循 “项 目 檬 板 目 录 + 控 制 占 名 称 前 级 ”的 命名 规范 《区 分 大 小 写 )。 所 以 create. 
html 文件 的 正确 文件 路 径 束 是 /app/Lib/View/Index/create.htm。 人 代码 如 下 。 


<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ 
TR/xhtml11/DTD/xhtml1-transitional.dtd"> 

<html xmlns="http://www.w3.0org/1999/xhtml1"> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 

seit le- Ié US HS ticle 

</head> 

<body> 

<form action="<!--{__APP__ }-->/index/post" method="post" name="form1"> 


<ul> 
[RA input ype tesi" name username! /S/n 
<1 > 密码 : <input type="password" name="password" /></li> 
</ul> 
<input Coe submit" value Wh mS 
</form> 
</body> 
</html> 


如 上 述 标 粗 代码 所 示 ， 表 示 将 表单 提交 到 当前 项 目 Index 控制 器 下 的 post 方法 。 在 该 方 
法 中 调用 Model 类 实现 数据 插入 操作 。 为 了 强化 模型 的 概念 ，CleverPHP 并 不 允许 直接 实例 
化 Model 基 类 ， 必 须 首 先 创建 实体 类 ， 并 继承 于 Model 基 类 。 自 定义 的 Model 类 名 需要 与 
数据 库 中 的 数据 表 名 称 相 对 应 (不 带 前 级 及 后 级 )。 所 以 ，user 表 的 实体 模型 就 是 
UserModel， 代 人 码 如 下 。 


<?php 
class UserModel extends Model? 


public function add(){ 
Şdata=array ( 
"username "=>mysql_escape_string($_POST["username"]), 
"password"=>md5 ($_POST["password"]), 
"add time"=>time () 


); 
Şrow=$this->data ($data)->create(); 
return $row; 
} 
} 


R 
0 Nee El e, R mHE post 动作 调用 即 可 。 代 码 如 下 。 
// 提 交 数 据 
public function Post () { 
if (empty ($_POST["username"])){ 
Sthis->message(" 用 户 名 不 能 为 空 ") ; 
}elseif (empty ($_POST["password"])){ 


Sthis->message ("密码 不 能 为 空 "); 
} 


$user=MOD ("user"); 
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if ($user->add()){ 
echo "搬入 成 功 "; 


}else{ 
echo "插入 失败 :" .Suser->error; 


} 

2. 查询 数据 

查询 数据 只 需要 调用 Mode 基 类 中 的 Read 方法 即 可 。 接 下 来 将 继续 以 前 面 创建 的 示例 
为 例 ， 在 UserModel 目 定 义 模 型 中 添加 奉 询 方法 。 如 以 下 代码 所 示 。 


/** 

* 显示 列表 

E 

public function rows (){ 
Ş$rows=$this->Read(); 


return $rows; 


} 
完成 后 ， 只 需要 在 控制 占 动 作 中 调用 ， 然 后 将 变量 值 分 配给 模板 引擎 即 可 。 代 人 码 如 下 。 
MAC 


public function Index (){ 


suser=MOD ("User"); 

Ş$rows=$user->rows(); 

Sthis->View(); 

$Sthis->smarty->assign ("pageTitle", gä DADNÉ Din: 
SE EE Ee ee Bee KE ow 
Şthis->smarty->display ("index.html"); 


} 

对 应 的 index.html 视图 模板 如 以 下 代码 所 示 。 

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" “http://www.w3.org/ 
TR/xhtml11/DTD/xhtml1-transitional.dtd"> 

<html xmlns="http://www.w3.0org/1999/xhtml1"> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 

<ritle><!=-={ pageTitle }==></ritle> 


</head><body> 
<UL> 
<!—-{foreach from=$list item=vo}--> 
<li><!--{$vo.username}--> <span><a href="<!--{ _ APP _}-->/Index/delete"> 删除 
</a></span><span><a href="<!--{ APP _}-->/Index/update"> 修 改 </a></span></1i><br> 
<!—-{/foreach}--> 
</ul></body> 
</html> 
删除 数据 


单 击 列表 中 的 “删除 ”链接 后 ， 将 进入 Delete 动作 ， 并 实现 数据 删除 操作 。 首 先 在 
UserModel 目 定 义 模型 中 添加 相应 的 删除 方法 ， 代 人 码 如 下 。 

/** 

* 删除 

SE 

public function delUser (){ 

return Sthils=>vrhere E EE ee 
} 
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控制 器 Delete 动作 代 人 如 下 。 


/ /删除 数据 
public function delete (){ 
TE (emocy le Kale ha 


Sthis->message ("参数 错误 ")， 


} 


$user=MOD ("user"); 


if ($user->delUser()){ 
Šthis->message (ler Uni: 
}else{ 
throw new Exception ("删除 失败 : " Şuser—->error); 


} 


} 
d. 更 新 数据 
单 击 “修改 ”链接 后 ， 将 进入 Update 页 面 ， 在 该 页 面 中 完成 数据 搜集 。Update 动作 代 


DIE, 
/ /更 新 数据 


public function Update () { 
suser=MOD ("user")}; 
ŞSrow=$user->where ("id=".intval($_GET["id"]))->find(); 


Sthis->View(); 
Şthis->smarty->assign("user", $row); 
Şthis->smarty->display ("update.html1"); 


} 

对 应 的 update bm) 视图 模板 代码 如 下 。 

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ 
TR/xhtml1/DTD/xhtml1l-transitional.dtd"> 

<html xmlns="http://www.w3.o0org/1999/xhtml1"> 


<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 


wele EI lr ele 


</head><body> 
<form action="<!--{__APP__}-—->/index/save/id/<!-—-{$user.id}-->" method="post" name= 
Een 
<ul> 
<li HPZ: <input type="test" value="<!-—-{Ş$user.username}-->" name="username" 
/EES LLS 


< input Cype password nane password -ii 
ll ERA <input type password" name repasswordn -</li> 


</ul> 
<input Eve "submit value Tia T > 
</form> 
</body> 
</html> 
HP REAREN, KHA Save 动作 ， 在 该 动作 中 完成 数据 更 狐 。 代 人 码 如 下 。 
/** 


* 提交 数据 更 新 
x Qthrows Exception 
人 


public function Save () { 
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suser=MOD ("user"); 


if (empty ($_POST["username"])){ 
sthis->message(" 用 户 名 不 能 为 空 ") ; 
}elseif (empty ($_POST ["passwordq" ] ) | | $5_POST["password"] !=$_POST["repassword"])t 


Sthis->message ("请 输入 正人 确 的 密码 ")， 
| 
sdata=array ( 
"username"=>mysql_escape_string($_ POST["username"]), 
"password"=>md5 ($_POST["password"]) 
E 


Li ($user—->whnere (Wie Sw, Lmeyval ($_ EBET [vicW]) -caca (Seara) ->tpcarce ()) 1 
Sthis->message ("数据 更 新 成 功 "); 
}else{ 


throw new Exception (" 数 据 更 新 失败 : ".$user->error); 


18.6.3 ”测试 驱动 


前 面 已 经 介绍 过 ，CleverPHP 的 Session 及 绥 存 驱动 是 使 用 中 间 件 实现 的 。 接 下 来 将 通 
过 代码 演示 Session 及 绥 存 驱动 的 使 用 。 

这 里 将 继续 使 用 Memcache 作为 Session 及 绥 存 数据 库 。 为 了 方便 操作 ， 这 里 将 Session 
及 绥 存 操作 分 别 封装 为 Session K S AZt. Session 图 数 如 以 下 代码 所 示 。 

Fees 


* session 


* Qparam string or int $name 
* @param string $value 
Ei 
function session ($name, $value=""){ 
if (empty(Sname) ) { 
return $_ El 
} 
$ssessino=new Session (); 
if (empty (Şvalue)){ 
return $sessino->get ($name); 
}elseif(is_null (Ş$value)){ 
return $sessino->clear ($name); 
}else{ 


return $sessino->set ($name, $value); 


} 

S 图 数 代码 如 下 。 

/ 大 大 

* 绥 存 快捷 函数 

* Qparam string $key 
x Qparam string S$data 
x Qparam int Sexpire 
GE 

neelon E EE ee ve Ee e Ee CHECK 
$sCache=new Cachet): 

if (empty (Şexpire)){ 


Şexpire=C ("memcache_expire"); 
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} 
if (lIempty ($data))f{ 

// 设 置 

return $Cache->set ($key, $data, $expire); 
}elseif(is_null ($data) ){ 

// 删 除 

return $Cache->delete (Skey Di: 
}else{ 

// 获 取 

return $Cache->get ($key); 
} 


} 
下 面 首 先 测试 Session 函数 ， 代 码 如 下 。 
public function Index (){ 
session ("username", "£F y}"); 
echo session ("username"); 
} 
要 确保 Session 操作 切换 到 SessionMemcache 驱动 ， 需 要 将 项 目 配置 文件 中 的 Session 配 


HAN memcache。 同 时 ， 需 要 确保 Memcache 处 理 正 常 运行 状态 。 
绥 存 的 使 用 同样 变 得 徐 单 ， 代 人 码 如 下 。 


public function Cache() { 


S$dqata=S("username" ) ，; 
te (lScerta) 
Sedara ta EE 
S ("username", $data); 


} 
echo $data; 


18.7 小结 


本 芋 结 合 了 全 书 内 容 特点 ， 循 序 渐 进 地 介绍 了 MVC 框架 的 设计 及 开发 过 程 。 本 革 内 容 
是 全 书 最 注 和 章 实 践 应 用 的 部 分 ， 不 要 求 读 者 全 部 掌握 ， 但 需要 读者 理解 并 熟悉 MVC IER 
计 的 流程 。 由 于 篇 幅 所 限 ， 在 还 有 大 量 代 码 及 类 库 并 没有 在 章节 里 进行 介绍 ， 需 要 该 者 结合 
本 书 配套 源码 在 实践 时 进行 深入 学 习 。 

开发 MVC 框架 是 需要 拉 术 及 思想 的 ， 但 这 些 都 需要 在 实践 中 进行 积累 ， 所 以 笔者 并 没 
有 以 堆砌 代码 的 形式 对 MVC 框 和 保 进 行 全 面 介绍 ， 而 是 从 一 开始 就 员 军 MVC 设计 思想 ， 选 
Hie WI CURD 操作 、 驱 动 扩 展 等 重点 内 容 进 行 介绍 ， 让 读者 领略 开 太 MVC 框 染 的 乐趣 。 
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附录 A 让 Nginx 支持 Pathinfo 模式 


提供 


Pathinfo 是 一 种 CGI 1.1 标准 ， 用 于 收集 文件 及 目录 路 径 信 息 。 大 多 数 PHP MVC 框架 都 


Pathinfo 访问 模式 ， 从 而 较 好 地 改善 用 户 体 验 。 对 于 Apache 而 言 ， 通 常情 况 下 不 需要 祝 


外 设置 ， 即 可 正常 使 用 。 但 是 对 于 Nginx 而 言 ， 安 装 完 成 后 并 不 能 使 用 Pathinfo， 这 无 疑 会 给 
我 们 在 开发 及 调试 MVC 应 用 时 ， 造 成 困扰 。 事 实 上 ，Nginx 可 以 通过 目 定 义 fastcgi_param SC 
现 对 Pathinfo 文 持 ， 接 下 来 将 以 Nginx 1.3.8 为 例 ， 详 细 介绍 Nginx 实现 对 Pathinfo LEF. 


件 。 


php.ini 是 否 已 经 开局 cgi.fix_pathinfo 文 持 ， 如 以 下 代码 所 示 。 


cgi.fix pathinfo provides *real* PATH_INFO/PATH_TRANSLATED support for CGI. PHP's 
previous behaviour was to set PATH_TRANSLATED to SCRIPT_FILENAME, and to not grok 
cais cO 1 will cause Ph? CEL tO ri ltrs paths cO conrcorm co tche spec, A Serccingj 

; what PATH_INFO is. For more information on PATH_INFO, see the cgi specs. Setting 
of zero causes PHP to behave as before. Default is 1. You should fix your scripts 

; to use SCRIPT_FILENAME rather than PATH_TRANSLATED. 

; http://php.net/cgi.fix-pathinfo 

cgi.fix_pathinfo=1 

将 cgi.fix_pathinfo 设置 为 1 即 可 。 完 成 后 重新 局 动 php-fpm， 然 后 打开 Nginx 配置 文 
找到 PHP 转发 配置 (位 于 server WA F), RIEU F- 


ee 
SE E pass 127:-0-0, 1:9000; 
fastcgi_index index.php; 
ser Şpatn inio v/v 
set Sreal_ script name sfastcgi_ script name; 
ii (Sfasteol seripr neame ~ WA( T? oN) (/-T)SY) 1 set Sreal scripete neame SL; 
set Spath info $2; 
} 


上 述 配置 信息 是 Nginx 1.3.8 默认 提供 的 PHP 转发 配置 。 接 下 来 需要 在 该 配置 信息 中 添 


加 


个 目 定义 fastcgi_param 项 ， 从 而 实现 Pathinfo 文 持 ， 代 人 码 如 下 。 


location ~ .+\.php { 
set S$script Suly 
set opath inilo T75 
set Sreal_ script name S$fastcgi_ script name; 
BEE | 
SEE SSCrIioE SLE 
ser Soden inito $27 
} 
e pass 127:0.:0, 1890007 
fastcgi index index.php?IF_REWRITE=1; 
include /usr/local/nginx/conf/fastcgi params; 
fastcgi param PATH INFO Spath info,; 
fastcgi_ Garam SCRIPT_FILENAME Sdocument root/$script,; 
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fastcgi_param SCRIPT_NAME S$script; 
fastcgi_param SCRIPT_NAME Ş$real_script_name; 
} 


通过 上 述 改动 ， 最 后 只 需要 保存 nginx.conf 配置 文件 ， 并 重启 Nginx， 即 可 实现 Pathinfo 
访问 支持 。 

需要 注意 的 是 ， 虽 然 通 过 配置 PHP 转发 能 够 实现 Pathinfo。 但 是 ， 完 整 模拟 隐藏 入 口 
Cindex.php) 文件 后 的 Pathinfo， 还 需要 配合 URL 重 写 功能 才能 实现 。 人 代码 如 下 。 


location / { 


index index.php; 


if (!-e Srequest filename) { 
rewrite MU /index:php/S1 last? 
break; 


} 


} 
上 述 转发 规则 很 简单 ， 作 用 就 是 将 所 有 以 “/” 结 尾 的 URL 请 求 都 重 写 到 index.php 文件 。 
当然 ， 读 者 也 可 以 将 重 写 规则 单独 配置 到 虚拟 主机 或 者 特定 的 项 目 目录 中 。 如 以 下 代码 所 未。 


location /tp/ { 
index index.php; 
if (le Srequest filename) { 
rewrite ^/tp/(.*)S /Eee/index ohne/sS last; 
break; 


} 


} 
至 此 ， 所 有 PHP MVC 框架 都 能 够 正常 使 用 Pathinfo 访问 模式 了 。 


HRB 配置 团队 开发 环境 


本 书 一 开始 就 介绍 了 SubVersion 的 实战 应 用 。 使 用 SubVersion， 可 以 让 团队 协作 变 得 简 
单 ， 很 好 地 解决 团队 开发 中 和 常见 的 问题 ， 例 如 代码 托管 、 代 码 备份 、 版 本 控制 等 。 但 这 里 有 
一 个 重要 的 问题 没有 解决 ， 那 束 是 团队 测试 。 

怎样 在 团队 开发 环境 中 测试 项 目 ， 这 是 普通 程序 员 所 经 常 遇 到 的 问题 。 和 常见 的 做 法 有 两 
种 : 一 种 是 使 用 本 地 环境 进行 调试 ， 然 后 通过 QQ, FTP 或 远程 管理 等 工具 传递 给 团队 其 他 
程序 员 ， 完 成 逐个 功能 测试 ， 另 一 种 是 使 用 网 络 文件 共享 。 前 者 效率 慢 ， 只 能 适用 于 2、3 
个 人 的 开发 环境 。 后 者 是 常用 的 大 型 团队 开发 测试 方案 ， 其 运行 原理 如 附 图 B-1 所 示 。 

附 图 B-1 所 示 ， 网 络 文件 共享 服务 将 Web 服务 器 〈 即 文件 服务 器 ) 上 的 文件 及 目录 了 映 
射 到 本 地 计算 机 上 ， 程 序 员 在 保存 及 测试 文件 时 ， 直 接 将 文件 保存 到 所 映射 的 目录 上 ， 就 相 
当 于 保存 到 Web 服务 器 上 。 这 一 过 程 ， 程 序 员 并 不 需要 对 文件 进行 任何 上 传 操作 ， 一 切 就 
如 同 本 地 文件 系统 一 样 方便 。 在 文件 被 保存 后 ， 团 队 内 的 其 他 成 员 可 以 实时 地 了 解 该 文件 开 
发 进度 ， 并 参与 测试 。 

实现 网 络 文件 共享 服务 的 套件 主要 有 CIFS (网 络 邻 居 ) 及 NFS。 前 者 主要 在 Windows 
操作 系统 下 使 用 ， 后 者 只 适用 于 Linux 或 UNIX 之 间 文 件 共 享 。 但 在 真实 环境 中 ， 通 常情 况 
下 都 是 服务 端 使 用 Linux BERR, mA mE Windows, Mac OS HAERA. XIT, Æ 
论 是 CIFS 或 者 NFS WBA REEK, MARA T AU HIH Samba 了 。 


E En 
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Web 服务 器 


程序 员 A 程序 员 B 程序 员 C 


附 图 B-1 网 络 文 件 共享 服务 


Samba 与 CIFS 及 NFS 一 样 ， 都 是 基于 SMB (Server Message Block) 通信 协议 实现 
的 。 不 同 之 处 在 于 Samba 不 仅 能 够 实现 Linux 之 间 进 行 网 络 文件 共享 ， 还 能 够 实现 Linux 与 
JE Linux (包括 Windows) 文件 共享 。 所 以 使 用 Samba 构建 团队 开发 环境 是 非常 明智 的 。 

由 于 Samba 非常 强大 及 易 用 ， 所 以 很 多 Linux 发 行 版 本 都 内 置 了 Samba 套件 。 需 要 说 
明 的 是 ， 虽 然 Samba 提供 了 可 靠 的 文件 共 孚 服务 ， 但 在 配置 团队 开发 环境 时 ， 需 要 注意 安全 
配置 。 此 外 ， 由 于 映射 的 目录 通 钊 为 Web 目录 ， 而 该 目录 所 属 组 及 用 户 通 党 归属 于 Web 服 
务 占 进程 。 而 正 是 这 个 原因 ， 导 伊 很 多 程序 员 在 配置 完成 并 局 动 Samba 后 ， 虽 然 可 以 正常 提 
供 网 络 文件 共 人 对 服务， 但 却 不 允许 文件 写 入 操作 。 这 无 疑 是 没有 意义 的 ， 接 下 来 将 通过 示例 
的 方式 ， 话 细 介 绍 使 用 Samba 构建 团队 开发 环境 的 全 过 程 。 

1. 安装 Samba 

假设 有 一 台 Linx (REX CentOS 6.0) Hz, IP 地 址 192.168.2.20。 现 在 使 用 该 服务 
右 作 为 团队 开发 文件 服务 占 。 为 了 方便 调试 程序 ， 访 服务 右上 已 经 安 疾 LNMP。 现 在 需要 安 
Ze Samba 文件 ， 以 实现 文件 共 孚 服务 。 步 骤 如 下 。 

EICI Samba 套件 。 

[root@localhost ~]# yum install cups-libs samba samba-common 

安装 完成 后 ， 将 会 在 /etc/samba/ 目 录 碍 看 到 Samba 配置 文件 。 


[root@localhost ~]# ls /etc/samba/ 


lmhosts smb.conf smbusers 
[root@localhost ~]# 
PIE, Samba 就 安装 完成 了 。 
2. 创建 共同 用 户 
利用 Samba 共享 目录 可 以 方便 地 搭建 远程 开发 服务 器 ， 我 们 在 本 地 储存 文件 时 ， 其 实 
操作 的 是 Linux 上 的 文件 系统 ， 但 效果 和 操作 本 地 文件 一 样 。 利 用 这 一 特点 ， 我 们 可 以 将 
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Linux 服务 右上 的 Web 目录 议 置 为 Samba 共享 目录 ， 并 且 将 共享 目录 及 Web 进程 拥有 者 设 
置 为 Samba 用 户 库 中 合法 的 用 户 ， 即 可 解决 权限 冲突 或 不 足 的 问题 。 步 又 如 下 。 


[root@localhost ~]# groupadd www 
[root@localhost ~]# useradd -g www www 

接 下 来 将 www 用 户 导 入 到 Samba IO "BC, 
[root@localhost ~]# smbpasswd -a www 


New SMB password: 
Retype new SMB password: 
[root@localhost ~]# 
上 上 述 操作 中 的 用 户 密码 并 不 是 Linux 系统 登录 和 密码， 而 是 Samba 用 户 数据 库 中 存放 的 密 
人 码 。 客 户 端 访问 共享 目录 时 需要 提供 该 密码 〈 查 看 已 有 Samba 用 户 可 以 使 用 pdbedit - L 命 
令 )。 完 成 后 ， 将 Nginx 的 进程 所 有 者 修改 为 www。 


[root@localhost ~]# vi /usr/local/nginx/conf/nginx.conf 


user WWW WWW} 
worker_processes l1; 
events { 

worker_connections 1024; 


} 

上 述 操 作 的 目的 是 确保 程序 员 在 本 地 操作 文件 时 ， 能 够 以 文件 所 有 者 的 权限 进行 。 完 成 
后 ， 只 和 需要 将 Samba 共享 目 采 定位 到 Nginx 网 站 主 目 录 即 可 。 

3. Be Gamba 

为 了 方便 介绍 ， 这 里 并 没有 对 Samba 详细 的 配置 进行 介绍 ， 有 需要 的 该 者 可 以 参考 相 
XK Linux 图 书 。 接 下 来 只 需要 配置 一 个 共享 目录 ， 即 可 实现 需求 。 


[root@localhost ~]# vi /etc/samba/smb.conf 


246 #============================= Share Definitions ============================== 
247 

248 [homes] 

249 comment = Home Directories 

250 browseable = yes 

251 writable = yes 

2 valid users = SZ 

2531 valid users = MYDOMAINASS 

254 [web] FEET 

255 comment = web smb HE 

256 path = /home/wwwroot #Nginx 网 站 主 目录 
257 writable = yes Wr 

258 browseable = yes # 是 否 可 浏览 

259 [DPFLMSES| 

260 comment = All Printers 

261 path = /var/spool/samba 

262 browseable = yes 

263 guest ok = yes 

264 writable = yes 

265 printable = yes 


全 此 ，Samba 束 配 置 完成 了 ， 接 下 来 内 需要 局 动 Samba 服务 即 可 完成 团队 开发 环境 
配置 。 
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4. 局 动 Samba 

STEI, F PRA AA Samba 服务 器 了 。 如 果 开 局 了 防火 场 ， 需 要 打开 139 
和 445 mHE CEH netstat min | grep smb 命令 查看 )。 同 时 ， 由 于 CentOS 6.x 默认 开局 
SELinux， 为 了 避免 给 实验 造成 困扰 ， 可 以 临时 将 其 关闭 。 

[root@localhost ~]# setenforce 0 

当然 ， 由 于 开发 服务 器 通常 都 是 处 于 内 网 的 ， 所 以 并 不 需要 较 高 的 安全 保护 。 我 们 也 可 
以 永久 关闭 SELinux。 


[root@localhost ~]# vi /etc/selinux/config 


#SELINUX=enforcing HEI 
#SELINUXTYPE=targeted # 注 释 
SELINUX=disabled # 增 加 


保存 后 ， 重 新 局 动 系统 即 可 生效 。 接 下 来 只 需要 局 动 Samba 服务 即 可 。 


[root@localhost ~]# service smb start 


[root@localhost ~]# chkconfig smb on 
5. 使 用 Samba 
Samba 启动 后 ， 接 下 来 束 可 以 在 客户 病 连 接 共享 目录 ， 从 而 实现 高 效 的 团队 协作 开 友 。 
在 Windows 7 中 的 步骤 如 下 。 
在 Windows 开始 亲 单 中 启动 “运行 ”对 话 框 ， 然 后 输入 Samba 服务 如 所 在 的 卫 地 址 
(本 例 Samba 服务 器 IP 地 址 为 192.168.2.20)， 如 附 图 B-2 所 示 。 


Windows 特 根 据 上 您 所 输入 的 名 称 ， 为 您 打开 相应 的 程序 ， 立 
EE, RERE Internet 资源 。 


打开 (D) IE 
E ”使 用 管理 权限 创建 此 任务 。 


附 图 B-2 Windows 打开 任务 对 话 框 
点 击 “ 人 确定 ”按钮 ， 输 入 www 认证 用 户 和 和 密码， 登录 成 功 后 ， 就 能 看 到 Web 网 站 下 的 
所 有 文件 了 。 些 时， 可 以 在 共享 目录 中 新 建 、 有 删除 、 修 改 等 常见 操作 ， 就 如 同 本 地 操作 一 
样 。 对 文件 进行 修改 并 保存 后 ， 可 以 通过 Web 地 址 访问 到 该 文件 。 为 了 便于 操作 ， 可 以 将 
共享 目录 映射 为 本 地 目录 ， 方 法 如 附 图 B-3 所 示 。 


要 映射 的 网 络 文 件 夹 : | 
EISE SEET GET 


X | DUSE. ， 
TA: VWerwoenchare 
v eRHEME R) 
WC E Ee ES tel 
mm ) en 


附 图 B-3 Dinant UA a 
至此， 团队 开发 环境 就 配置 完成 了 。 
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4y IMC A tok) o ÆA, DEET EE EE EE 只 要 读者 掌 
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车 ， 唯 一 需要 的 是 读者 人 六 理解 MTVGC 设 计 思 想 妈 可 。 对 于 这 一 点 ， 夺 懈 的 前 小 部 分 已 红 使 用 了 
大 重 的 素 例 进行 了 讲解 。 所 沙 无 论 你 是 哪个 阶段 的 Vep 从 业 人 员 ， 计 条 都 是 运 合 你 的 。 

桔 属 并 不 是 一 车 PHP MVC ESMA, zë SAAR SA APLATI AKRA 
Se bThinkPHOP A Zi zc oi" Sigk ga ZE RA Aa ST ZSA, 
habe ZE EE ECH E EE ckt CEET EE EE 
更 好 地 理解 和 使 用 /PHP MTVYC 开 发 模式 。 

当 铸 ， 无 论 学 习 哪 门 技术 ， 都 是 证 苦 和 快 东 的 。 与 广大 读者 一 样 ， 笔 者 也 是 一 位 长 年 从 事 
gx 互联 网 相关 工作 的 开发 人 页 ， 学 知 gC 从 业 信 员 的 苦 占 东 。 如 果 你 在 学 习 过 程 中 遇 到 力 难 ， 
请 不 要 语 党 求助 无 门 ， 你 可 履 让 书 约 前 言 中 找到 我 的 文 流言 次 ， 我 会 研 必 赔 读 你 鬼 诉 说 ， 并 会 
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